HDU5286 wyh2000 and sequence

题意:

给出长为\(N\)的序列\(A_1,A_2,A_3,\cdots,A_n\),\(q\)次询问,每次询问给出区间\([L,R]\),假设区间内出现过的数为\(C_1,C_2,\cdots,C_k\),出现的次数分别为\(B_1,B_2,\cdots,B_k\),输出\(\sum_{i=1}^{k}C_i^{B_i} % 1000000007\),要求在线

题解:

如果不要求在线,直接用莫队很快就能解决,现在要求在线,还是考虑把序列分块,预处理出所有连续的块的答案,和到第\(i\)块为止,各个数出现的次数的前缀和

对于每次查询,如果在一个块内,就直接暴力查询

否则对于在两端非整块的元素,记录元素大小和出现次数,将这些元素带来的贡献计算一下即可

有点卡常

//#pragma GCC optimize("O3")
//#pragma comment(linker, "/STACK:1024000000,1024000000")
#include<bits/stdc++.h>
using namespace std;
function<void(void)> ____ = [](){ios_base::sync_with_stdio(false); cin.tie(0); cout.tie(0);};
const int MOD = 1e9+7;
const int MAXN = 5e4+7;
const int SQT = 240;
int n,q,m,sqt,A[MAXN],pre[SQT][MAXN],pos[MAXN],belong[MAXN],l[SQT],r[SQT],la;
int ans[SQT][SQT],num[MAXN];
vector<int> vec,cnt,pw[MAXN];
void divide(){
sqt = pow(n,0.5);
m = n / sqt + (n%sqt?1:0);
for(int i = 1; i <= n; i++) belong[i] = (i-1) / sqt + 1;
for(int i = 1; i <= m; i++) l[i] = (i-1) * sqt + 1, r[i] = i * sqt;
r[m] = n;
}
void gao(){
int L, R;
scanf("%d %d",&L,&R);
L = (L^la)%n+1; R = (R^la)%n+1;
if(L>R) L ^= R ^= L ^= R;
int lp = belong[L], rp = belong[R];
if(lp==rp){
int ret = 0;
for(int i = L; i <= R; i++) num[pos[i]] = 0;
for(int i = L; i <= R; i++){
ret = ret - pw[pos[i]][num[pos[i]]];
ret = ret + pw[pos[i]][++num[pos[i]]];
if(ret<0) ret += MOD;
if(ret>=MOD) ret -= MOD;
}
printf("%d\n",la=ret);
}
else{
vector<int> arr;
for(int i = L; i <= r[lp]; i++){
num[pos[i]] = 0;
arr.push_back(pos[i]);
}
for(int i = l[rp]; i <= R; i++){
num[pos[i]] = 0;
arr.push_back(pos[i]);
}
sort(arr.begin(),arr.end()); arr.erase(unique(arr.begin(),arr.end()),arr.end());
for(int i = L; i <= r[lp]; i++) num[pos[i]]++;
for(int i = l[rp]; i <= R; i++) num[pos[i]]++;
lp++, rp--;
int ret = ans[lp][rp];
for(int i = 0; i < (int)arr.size(); i++){
ret = ret - pw[arr[i]][pre[rp][arr[i]]-pre[lp-1][arr[i]]];
ret = ret + pw[arr[i]][pre[rp][arr[i]]-pre[lp-1][arr[i]]+num[arr[i]]];
if(ret<0) ret += MOD;
if(ret>=MOD) ret -= MOD;
}
printf("%d\n",la=ret);
}
}
void solve(){
scanf("%d %d",&n,&q);
vec.clear();
for(int i = 1; i <= n; i++){
scanf("%d",&A[i]);
vec.push_back(A[i]);
}
sort(vec.begin(),vec.end()); vec.erase(unique(vec.begin(),vec.end()),vec.end());
for(int i = 1; i <= n; i++) pos[i] = lower_bound(vec.begin(),vec.end(),A[i]) - vec.begin();
divide();
for(int i = 1; i <= m; i++) for(int j = 0; j < (int)vec.size(); j++) pre[i][j] = 0;
for(int i = 1; i <= n; i++) pre[belong[i]][pos[i]]++;
for(int i = 1; i <= m; i++) for(int j = 0; j < (int)vec.size(); j++) pre[i][j] += pre[i-1][j];
cnt.resize(vec.size());
for(int i = 0; i < (int)vec.size(); i++) cnt[i] = 0;
for(int i = 1; i <= n; i++) cnt[pos[i]]++;
for(int i = 0; i < (int)vec.size(); i++){
pw[i].resize(cnt[i]+1);
pw[i][0] = 1; for(int j = 1; j <= cnt[i]; j++) pw[i][j] = 1ll * pw[i][j-1] * vec[i] % MOD;
pw[i][0] = 0;
}
for(int i = 1; i <= m; i++){
memset(num,0,sizeof(num));
int p = l[i] - 1, ret = 0;
for(int j = i; j <= m; j++){
while(p<r[j]){
p++;
ret -= pw[pos[p]][num[pos[p]]];
ret += pw[pos[p]][++num[pos[p]]];
if(ret<0) ret += MOD;
if(ret>=MOD) ret -= MOD;
}
ans[i][j] = ret;
}
}
la = 0;
while(q--) gao();
for(int i = 0; i < (int)vec.size(); i++) pw[i].clear();
}
int main(){
int T;
for(scanf("%d",&T); T; T--) solve();
return 0;
}

HDU5286 wyh2000 and sequence【分块 均摊复杂度】的更多相关文章

  1. Mr. Kitayuta's Colorful Graph CodeForces - 506D(均摊复杂度)

    Mr. Kitayuta has just bought an undirected graph with n vertices and m edges. The vertices of the gr ...

  2. cf250D. The Child and Sequence(线段树 均摊复杂度)

    题意 题目链接 单点修改,区间mod,区间和 Sol 如果x > mod ,那么 x % mod < x / 2 证明: 即得易见平凡, 仿照上例显然, 留作习题答案略, 读者自证不难. ...

  3. 3L-最好、最坏、平均、均摊时间复杂度

    关注公众号 MageByte,设置星标点「在看」是我们创造好文的动力.后台回复 "加群" 进入技术交流群获更多技术成长. 本文来自 MageByte-青叶编写 上次我们说过 时间复 ...

  4. 洛谷 P6783 - [Ynoi2008] rrusq(KDT+势能均摊+根号平衡)

    洛谷题面传送门 首先显然原问题严格强于区间数颜色,因此考虑将询问离线下来然后用某些根号级别复杂度的数据结构.按照数颜色题目的套路,我们肯定要对于每种颜色维护一个前驱 \(pre\),那么答案可写作 \ ...

  5. Chapter4 复杂度分析(下):浅析最好,最坏,平均,均摊时间复杂度

    四个复杂度分析: 1:最好情况时间复杂度(best case time complexity) 2:最坏情况时间复杂度(worst case time complexity) 3:平均情况时间复杂度( ...

  6. 【loj6029】「雅礼集训 2017 Day1」市场 线段树+均摊分析

    题目描述 给出一个长度为 $n$ 的序列,支持 $m$ 次操作,操作有四种:区间加.区间下取整除.区间求最小值.区间求和. $n\le 100000$ ,每次加的数在 $[-10^4,10^4]$ 之 ...

  7. 【uoj#228】基础数据结构练习题 线段树+均摊分析

    题目描述 给出一个长度为 $n$ 的序列,支持 $m$ 次操作,操作有三种:区间加.区间开根.区间求和. $n,m,a_i\le 100000$ . 题解 线段树+均摊分析 对于原来的两个数 $a$ ...

  8. HDU - 6098:Inversion(暴力均摊)

    Give an array A, the index starts from 1. Now we want to know B i =max i∤j A j  Bi=maxi∤jAj , i≥2 i≥ ...

  9. 《数据结构与算法之美》 <02>复杂度分析(下):浅析最好、最坏、平均、均摊时间复杂度?

    上一节,我们讲了复杂度的大 O 表示法和几个分析技巧,还举了一些常见复杂度分析的例子,比如 O(1).O(logn).O(n).O(nlogn) 复杂度分析.掌握了这些内容,对于复杂度分析这个知识点, ...

随机推荐

  1. Angular入门到精通系列教程(7)- 组件(@Component)基本知识

    1. 概述 2. 创建Component 组件模板 视图封装模式 特殊的选择器 :host inline-styles 3. 总结 环境: Angular CLI: 11.0.6 Angular: 1 ...

  2. 笔记:学习go语言的网络基础库,并尝试搭一个简易Web框架

    在日常的 web 开发中,后端人员常基于现有的 web 框架进行开发.但单纯会用框架总感觉不太踏实,所以有空的时候还是看看这些框架是怎么实现的会比较好,万一要排查问题也快一些. 最近在学习 go 语言 ...

  3. 【JDBC核心】批量插入

    批量插入 批量执行 SQL 语句 当需要成批插入或者更新记录时,可以采用 Java 的批量更新机制,这一机制允许多条语句一次性提交给数据库批量处理.通常情况下比单独提交处理更有效率. JDBC 的批量 ...

  4. servlet+jsp完成简单登录

    将用户在注册界面中的数据填充到数据库相对应的表格中.当用户再次登录时,从数据库中拿到相应的数据查询并与页面的数据做对比,判断是否登陆成功. 需要在HTML文件中将form表单上的action属性值设置 ...

  5. 虚拟机linux共享文件夹

    linux共享文件夹问题 1. 初配置 2. 挂载 018.7.8 为了实现win7与VMware中linux文件夹共享很花费了一些时间,但终成正果 1. 初配置 虚拟机设置->选项->共 ...

  6. qmake奇淫技巧之字符串宏定义

    阅读本文大概需要3.3分钟 我们平时在软件开发过程中需要定义一些宏,以便在代码中调用,这样每次不需要修改代码,只需要修改外部编译命令就可以得到想要的参数,非常方便 比如我们想在软件介绍中显示软件版本, ...

  7. 使用Intelij 运行Android 程序导致的无法安装

    前几天的时候更换了开发工具开发Android ,终于不用忍受Android studio 的各种卡顿了.我决定使用一段时间Intelij 开发Android. 之前的程序代码在运行的时候也出现了异常, ...

  8. 基于.NET Core的优秀开源项目合集

    开源项目非常适合入门,并且可以作为体系结构参考的好资源, GitHub中有几个开源的.NET Core项目,这些项目将帮助您使用不同类型的体系结构和编码模式来深入学习 .NET Core技术, 本文列 ...

  9. centos7 开放指定端口

    centos7 开放指定端口 #开放8080端口 firewall-cmd --zone=public --add-port=8080/tcp --permanent #重载防火墙 firewall- ...

  10. HTML基础复习1

    网页:HTML(超文本标记语言) 网页分为静态网页和动态网页,区别:动态网页中可以加入脚本代码,还可以动态的引入数据库中的信息. HTML的结构 <html> <head>头信 ...