Codeforces Round #525 (Div. 2) C. Ehab and a 2-operation task(数论构造)

思路来源

http://www.cnblogs.com/Dup4/p/10068891.html

题解

注意%可以起到等同于减的作用。

Solution1:

先给区间[1,n]加上一个较大的数D,比如500W,

这样ai就变为D+ai,

由于ai-i<1e5,此时再对[1,i]mod (D+ai-i)时,

D+ai就变为i了,而所mod数D+ai-i大于上一个数i-1,

即(i-1)%(D+ai-i)=i-1

也就是说对[1,i]区间操作时,[1,i-1]不会受到影响。

Solution2:

特判n=1直接输出0 辣鸡就是这么被hack掉的

n>1的情况,

第1次:

先对区间[1,n]加上一个数n,这样最后一个数不会小于n

第2到n-1次:

我们令前n-1个数分别为1,…,n-1即可,

计从后到前的已累加数为lazy,

由于第i个数最后要mod n变为i

故(ai+lazy+b)%n==i,ai+lazy+b==i+kn,

b==i-ai-lazy+kn,即令b=((i-ai-lazy)%n+n)%n,

对当前数加b,然后lazy+=b,

第n次:

对区间[1,n-1]mod n

代码

#include <iostream>
#include <algorithm> 
#include <cstring>
#include <cstdio>
#include <cmath>
#include <set>
#include <map>
#include <vector>
#include <stack>
#include <queue>
#include <functional>
const int INF=0x3f3f3f3f;
const int maxn=1e5+10; 
const int mod=1e9+7;
const int MOD=998244353;
const double eps=1e-7;
typedef long long ll;
#define vi vector<int> 
#define si set<int>
#define pii pair<int,int> 
#define pi acos(-1.0)
#define pb push_back
#define mp make_pair
#define lowbit(x) (x&(-x))
#define sci(x) scanf("%d",&(x))
#define scll(x) scanf("%lld",&(x))
#define sclf(x) scanf("%lf",&(x))
#define pri(x) printf("%d",(x))
#define rep(i,j,k) for(int i=j;i<=k;++i)
#define per(i,j,k) for(int i=j;i>=k;--i)
#define mem(a,b) memset(a,b,sizeof(a)) 
using namespace std;
int n,a[maxn];
ll pre;
int main()
{ 
   sci(n);
   rep(i,1,n)sci(a[i]);
   if(n==1)puts("0");
   else
   {
   printf("%d\n",n+1);
   printf("%d %d %d\n",1,n,n);//最后一个数大于等于n 
   pre=n;
   for(ll i=n-1;i>=1;--i)
   {
   	ll tmp=pre+a[i],tmp2=((i-tmp)%n+n)%n;
   	printf("%lld %lld %lld\n",(ll)1,i,tmp2);
   	pre+=tmp2;
   }
   printf("%d %d %d\n",2,n-1,n);//前n-1个数mod n  
   }
   return 0;
}

猜你喜欢

转载自blog.csdn.net/Code92007/article/details/84816051