【题意】
1 把第K个点加上D;
2 把区间[l,r]里的数变为离它最近的斐波那契数(这个fib数最小,且这个fib减这个数的绝对值也最小)
3 询问[l,r]区间内的和
【解题方法】维护两个sum。sum1存放当前区间的和,sum2存放当前区间所有数变成fib之后的和。这样,进行覆盖的时候,被覆盖的区间直接sum1=sum2就行了。
【AC 代码】
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <iostream>
using namespace std;
const int maxn=100010;
#define ll __int64
ll fib[100];
struct node{
int l,r;
ll sum1,sum2;//sum1存放当前区间和,sum2存放变成fib数之后的区间和
int lazy;
}Tree[maxn*4];
void pushup(int rt)
{
Tree[rt].sum1=Tree[rt*2].sum1+Tree[rt*2+1].sum1;
Tree[rt].sum2=Tree[rt*2].sum2+Tree[rt*2+1].sum2;
}
void pushdown(int rt)
{
if(Tree[rt].lazy){
Tree[rt*2].lazy=Tree[rt*2+1].lazy=1;
Tree[rt*2].sum1=Tree[rt*2].sum2;
Tree[rt*2+1].sum1=Tree[rt*2+1].sum2;
Tree[rt].lazy=0;
}
}
ll getval(ll x)
{
int pos=lower_bound(fib,fib+90,x)-fib;
if(pos&&abs(x-fib[pos-1])<=abs(x-fib[pos])) return fib[pos-1];
else return fib[pos];
}
void Build(int l,int r,int rt)
{
Tree[rt].l=l,Tree[rt].r=r;
Tree[rt].lazy=0;
if(l==r){
Tree[rt].sum1=0;
Tree[rt].sum2=1;
return ;
}
int mid=(l+r)/2;
Build(l,mid,rt*2);
Build(mid+1,r,rt*2+1);
pushup(rt);
}
void add(int pos,int val,int rt)
{
if(Tree[rt].l==pos&&Tree[rt].r==pos){
Tree[rt].sum1+=val;
Tree[rt].sum2=getval(Tree[rt].sum1);
return ;
}
pushdown(rt);
int mid=(Tree[rt].l+Tree[rt].r)/2;
if(pos<=mid) add(pos,val,rt*2);
else add(pos,val,rt*2+1);
pushup(rt);
}
void update(int L,int R,int rt)
{
if(L==Tree[rt].l&&Tree[rt].r==R){
Tree[rt].sum1=Tree[rt].sum2;
Tree[rt].lazy=1;
return ;
}
pushdown(rt);
int mid=(Tree[rt].l+Tree[rt].r)/2;
if(R<=mid) update(L,R,rt*2);
else if(L>mid) update(L,R,rt*2+1);
else{
update(L,mid,rt*2);
update(mid+1,R,rt*2+1);
}
pushup(rt);
}
ll queryans(int L,int R,int rt)
{
if(L==Tree[rt].l&&Tree[rt].r==R){
return Tree[rt].sum1;
}
pushdown(rt);
int mid=(Tree[rt].l+Tree[rt].r)/2;
if(R<=mid) return queryans(L,R,rt*2);
else if(L>mid) return queryans(L,R,rt*2+1);
else{
return queryans(L,mid,rt*2)+queryans(mid+1,R,rt*2+1);
}
}
int main()
{
fib[0]=1LL;
fib[1]=1LL;
for(int i=2; i<=90; i++) fib[i]=fib[i-1]+fib[i-2];
int n,m;
while(scanf("%d%d",&n,&m)!=EOF)
{
Build(1,n,1);
int id,l,r;
while(m--)
{
scanf("%d%d%d",&id,&l,&r);
if(id==1) add(l,r,1);
else if(id==2) printf("%I64d\n",queryans(l,r,1));
else update(l,r,1);
}
}
}