题目描述

题解:

坑题搞了三天。

莫队+线段树。

还有一些和斐波那契数列有关的性质。

首先答案是$a_1f_1+a_2f_2+…+a_nf_n$,

考虑插进去一个元素对答案产生的影响。

比如插进去一个$a_0$,插进去之后会排到第$k$位。

那么答案是$a_1f_1+a_2f_2+…+a_0f_k+a_kf_{k+1}+…+a_nf_{n+1}$,

等于$ans0+a_0f_k+a_kf_{k-1}+…+a_nf_{n-1}$。

所以单点插入,区间加斐波那契数?

不可能的。

考虑用矩阵推出斐波那契数列。

其实不需要矩阵快速幂,存当前$a_i*f_i$和前一个数$a_i*f_{i-1}$即可,

比如我们存的是$(a,b)$,那么有

$(a,b)->(a+b,a)->(2a+b,a+b)->(3a+2b,2a+b)->(5a+3b,3a+2b)->……$

系数都是斐波那契数。

向前转移同理

$(a,b)->(b,a-b)->(a-b,-a+2b)->(-a+2b,2a-3b)->……$

这个系数可以和斐波那契数列一起求,转移也很好推。

线段树单点插入时要一起搞事情,取模要少取,不然会$T$的很惨。

代码:

#include<cmath>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
const int N = ;
template<typename T>
inline void read(T&x)
{
T f = ,c = ;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){c=c*+ch-'';ch=getchar();}
x = f*c;
}
int n,qq,w[N],to[N],m=,B=;
int MOD,f[N],g[N],ans[N];
void Mod(int&x){if(x>=MOD)x-=MOD;}
void init()
{
if(MOD==)return ;
f[]=f[]=;
for(int i=;i<=n+;i++)
Mod(f[i]=f[i-]+f[i-]);
g[]=,g[]=MOD-;
for(int i=;i<=n+;i++)
Mod(g[i]=g[i-]-g[i-]+MOD);
}
struct Pair
{
int x,y;
Pair(){}
Pair(int x,int y):x(x),y(y){}
}p[N];
bool cmp(Pair a,Pair b){return a.x<b.x;}
struct Q
{
int l,r,id;
Q(){}
Q(int l,int r,int i):l(l),r(r),id(i){}
}q[N];
bool CMP(Q a,Q b){return (a.l/B)==(b.l/B)?(a.r<b.r):(a.l<b.l);}
struct segtree
{
int siz[N<<],tag[N<<];
int c[N<<][];
void update(int u)
{
siz[u] = siz[u<<]+siz[u<<|];
Mod(c[u][] = c[u<<][]+c[u<<|][]);
Mod(c[u][] = c[u<<][]+c[u<<|][]);
}
void add(int u,int k)
{
tag[u] += k;
int a = c[u][],b = c[u][];
if(k>)
{
(c[u][] = a*f[k+]+b*f[k])%=MOD;
(c[u][] = a*f[k]+b*f[k-])%=MOD;
}else if(k<)
{
k = -k;
(c[u][] = a*g[k-]+b*g[k])%=MOD;
(c[u][] = a*g[k]+b*g[k+])%=MOD;
}
}
void pushdown(int u)
{
if(tag[u])
{
add(u<<,tag[u]);
add(u<<|,tag[u]);
tag[u] = ;
}
}
void insert(int l,int r,int u,int qx,int k,int d)
{
if(l==r)
{
if(k==-)c[u][]=c[u][]=,siz[u] = ;
else c[u][] = f[k]*to[l]%MOD,c[u][] = f[k-]*to[l]%MOD,siz[u] = ;
return ;
}
pushdown(u);
int mid = (l+r)>>;
if(qx<=mid)insert(l,mid,u<<,qx,k,d),add(u<<|,d);
else insert(mid+,r,u<<|,qx,k+(k!=-)*siz[u<<],d);
update(u);
}
}tr;
int vis[N];
inline void push(int x)
{
if(!x)return ;
if(!vis[x])tr.insert(,m,,x,,);
vis[x]++;
}
inline void pull(int x)
{
if(!x)return ;
vis[x]--;
if(!vis[x])tr.insert(,m,,x,-,-);
}
int main()
{
// freopen("tt.in","r",stdin);
read(n),read(MOD);
init();
for(int a,i=;i<=n;i++)
read(a),p[i]=Pair(a,i);
sort(p+,p++n,cmp);
for(int las = 0x3f3f3f3f,i=;i<=n;i++)
{
if(las != p[i].x)
{
las = p[i].x;
to[++m] = las%MOD;
}
w[p[i].y] = m;
}
read(qq);
for(int l,r,i=;i<=qq;i++)
{
read(l),read(r);
q[i]=Q(l,r,i);
}
sort(q+,q++qq,CMP);
int L = ,R = ;
for(int i=;i<=qq;i++)
{
while(R<q[i].r)push(w[++R]);
while(R>q[i].r)pull(w[R--]);
while(L<q[i].l)pull(w[L++]);
while(L>q[i].l)push(w[--L]);
ans[q[i].id]=tr.c[][];
}
for(int i=;i<=qq;i++)
printf("%d\n",ans[i]);
return ;
}

CF633H Fibonacci-ish II的更多相关文章

  1. 【CF633H】Fibonacci-ish II 莫队+线段树

    [CF633H]Fibonacci-ish II 题意:给你一个长度为n的序列$a_i$.m个询问,每个询问形如l,r:将[l,r]中的所有$a_i$排序并去重,设得到的新数列为$b_i$,求$b_1 ...

  2. CF633H Fibonacci-ish II 莫队、线段树、矩阵乘法

    传送门 这题除了暴力踩标程和正解卡常数以外是道很好的题目 首先看到我们要求的东西与\(Fibonacci\)有关,考虑矩阵乘法进行维护.又看到\(n \leq 30000\),这告诉我们正解算法其实比 ...

  3. 【TOJ 3600】Fibonacci II (对数+斐波那契通项式)

    描述 2007年到来了.经过2006年一年的修炼,数学神童zouyu终于把0到100000000的Fibonacci数列(f[0]=0,f[1]=1;f[i] = f[i-1]+f[i-2](i> ...

  4. CF633H Fibonacci-ish II(莫队+线段树)

    温馨提示:本题十分卡常数,我手动开O2才过的.而数据范围不伦不类的n<=30000,常数小的O(n2)居然比O(n√nlogn)跑得快…… 考虑插进去一个元素对答案产生的影响.原本数列为Σa[i ...

  5. ZOJ3774_Power of Fibonacci

    求fibonacci数列前N个数的K次方和. 通项公式:F[n]=((1+sqrt(5))/sqrt(5)-(1-sqrt(5))/sqrt(5))/sqrt(5). 有点乱,不过由于可以保证最后的结 ...

  6. poj3070 Fibonacci

    Description In the Fibonacci integer sequence, F0 = 0, F1 = 1, and Fn = Fn − 1 + Fn − 2 for n ≥ 2. F ...

  7. 1143 多少个Fibonacci数

    时间限制:500MS  内存限制:65536K提交次数:270 通过次数:16 题型: 编程题   语言: C++;C Description 给你如下Fibonacci 数的定义: F1 = 1 F ...

  8. Codeforces Gym 100286F Problem F. Fibonacci System 数位DP

    Problem F. Fibonacci SystemTime Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hust.edu.cn/vjudg ...

  9. LeetCode: Linked List Cycle II 解题报告

    Linked List Cycle II Given a linked list, return the node where the cycle begins. If there is no cyc ...

随机推荐

  1. 20170406-ms

    Interval 间隔 revoke v撤销 alert adj 警觉的 n警报

  2. bzoj 4504: K个串【大根堆+主席树】

    像超级钢琴一样把五元组放进大根堆,每次取一个出来拆开,(d,l,r,p,v)表示右端点为d,左端点区间为(l,r),最大区间和值为v左端点在p上 关于怎么快速求区间和,用可持久化线段树维护(主席树?) ...

  3. bzoj 4197: [Noi2015]寿司晚宴【状压dp】

    一个数内可能多个的质因数只有小于根号n的,500内这样的数只有8个,所以考虑状压 把2~n的数处理出小于根号500的质因数集压成s,以及大质数p(没有就是1),然后按p排序 根据题目要求,拥有一个质因 ...

  4. github最值得收藏的Bootstrap3后台管理框架

    1. AdminLTE AdminLTE是一个完全响应的后台管理模板.基于Bootstrap3框架.高度可定制,易于使用.适合许多屏幕分辨率从小型移动设备到大型台式机. GitHub AdminLTE ...

  5. c++关键字explicit

    关键字explicit,可以阻止不应该允许的经过转换构造函数进行的隐式转换的发生.声明为explicit的构造函数不能在隐式转换中使用. C++中, 一个参数的构造函数(或者除了第一个参数外其余参数都 ...

  6. Hdu 4513 吉哥系列故事——完美队形II (manacher变形)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4513 题目描述: 打完题目描述了,点开题目,发现题目是中文,orz.jpg.果断又删掉了,习惯真可怕 ...

  7. stack(单调栈) POJ 2082 Terrible Sets

    题目传送门 题意:紧贴x轴有一些挨着的矩形,给出每个矩形的长宽,问能组成的最大矩形面积为多少 分析:用堆栈来维护高度递增的矩形,遇到高度小的,弹出顶部矩形直到符合递增,顺便计算矩形面积,且将弹出的宽度 ...

  8. 使用JS移除select的某些选项

    var arrvalue = new Array("1", "3", "4", "5", "6", ...

  9. sql server 2012 从删库到跑路

    问题: 向sql server 2012单个数据库中导入1500万+条数据的时候,报错: 错误 0xc0202009: 数据流任务 1: SSIS 错误代码 DTS_E_OLEDBERROR.出现 O ...

  10. 学习Python的day1

    自己以前从来没有写博客的想法,但是学Python,里面的老师也说了,写博客可以加深自己的记忆,也能回顾内容.还能给别人参考.挺值的.2017-09-16 一. Python介绍 python的创始人为 ...