首先嘛这道题目只要知道一个东西就很容易了:所有循环的最小公约数<=60,成一条链的长度最大为11,那么我们就可以用一个很裸的方法。对于在链上的数,我们修改直接暴力找出并修改。对于在环上的数,我们对每一步建立一颗线段树,那么修改就变成了交换60棵线段树的某个子树。然后我们就可以愉快的写啦~~~

在想的时候被同学坑了,他说最长的循环不超过60.。。。

在实现方面,我用一个map+树状数组维护链上的答案,然后用60棵线段树来维护这个环。懂了之后还是很好写哒~~~

Code:

 #include <cstdio>
#include <cstring>
#include <algorithm> const int N = + , kCycle = + ; int n, m, p, x[N]; int dist[N], cycle = ; inline int GCD(int a, int b) { return b ? GCD(b, a % b) : a; }
inline int LCM(int a, int b) { return a * b / GCD(a, b); } int pre[N]; void Analysis(int a) {
dist[a] = -;
int b = a * a % p;
if (dist[b] == -) {
pre[b] = a;
Analysis(b);
} else if (dist[b] == -) {
int cnt = ;
for (int i = a; i != b; i = pre[i], ++cnt) dist[i] = ;
dist[b] = ;
cycle = LCM(cycle, cnt);
}
if (dist[a] == -) dist[a] = dist[b] + ;
} inline int pos(int l, int r) { return (l + r) | (l != r); } int tree[N * ][kCycle], shift[N * ]; inline void Merge(int u[], int a[], int b[]) {
for (int i = ; i < cycle; ++i) u[i] = a[i] + b[i];
} inline void Shift(int id, int v) {
static int temp[kCycle * ];
(shift[id] += v) %= cycle;
for (int i = ; i < cycle; ++i) temp[i] = tree[id][(i + v) % cycle];
std::copy(temp, temp + cycle, tree[id]);
} void Modify(int l, int r, int u, int v) {
int id = pos(l, r);
for (int i = ; i < cycle; ++i, (v *= v) %= p) tree[id][i] += v;
if (l == r) return;
int mid = (l + r) / ;
if (shift[id]) {
Shift(pos(l, mid), shift[id]);
Shift(pos(mid + , r), shift[id]);
shift[id] = ;
}
if (u <= mid) Modify(l, mid, u, v); else Modify(mid + , r, u, v);
} void Cycle(int l, int r, int u, int v) {
int id = pos(l, r);
if (u <= l && r <= v) {
Shift(id, );
return;
}
int mid = (l + r) / ;
if (shift[id]) {
Shift(pos(l, mid), shift[id]);
Shift(pos(mid + , r), shift[id]);
shift[id] = ;
}
if (u <= mid) Cycle(l, mid, u, v);
if (v > mid) Cycle(mid + , r, u, v);
Merge(tree[id], tree[pos(l, mid)], tree[pos(mid + , r)]);
} int Query(int l, int r, int u, int v) {
int id = pos(l, r);
if (u <= l && r <= v) return tree[id][];
int mid = (l + r) / , res = ;
if (shift[id]) {
Shift(pos(l, mid), shift[id]);
Shift(pos(mid + , r), shift[id]);
shift[id] = ;
}
if (u <= mid) res += Query(l, mid, u, v);
if (v > mid) res += Query(mid + , r, u, v);
return res;
} class BIT {
int data[N]; public:
inline void Add(int p, int v) { for (; p <= n; p += p & -p) data[p] += v; }
inline int Query(int p) {
int res = ;
for (; p; p ^= p & -p) res += data[p];
return res;
}
inline int Query(int l, int r) { return Query(r) - Query(l - ); } } cnt, sum; void Sqr(int l, int r) {
if (!cnt.Query(l, r)) return;
if (l == r) {
cnt.Add(l, -);
sum.Add(l, -x[l]);
if (cnt.Query(l, l) == )
Modify(, n, l, (x[l] *= x[l]) %= p);
else
sum.Add(l, (x[l] *= x[l]) %= p);
} else {
int mid = (l + r) / ;
Sqr(l, mid);
Sqr(mid + , r);
}
} int main() {
scanf("%d%d%d", &n, &m, &p);
for (int i = ; i <= n; ++i) scanf("%d", x + i);
memset(dist, -, sizeof dist);
for (int i = ; i < p; ++i) if (dist[i] == -) Analysis(i);
for (int i = ; i <= n; ++i) {
if (dist[x[i]]) {
sum.Add(i, x[i]);
cnt.Add(i, dist[x[i]]);
} else {
Modify(, n, i, x[i]);
}
}
while (m--) {
int op, l, r;
scanf("%d%d%d", &op, &l, &r);
switch (op) {
case :
Cycle(, n, l, r);
Sqr(l, r);
break;
case :
printf("%d\n", Query(, n, l, r) + sum.Query(l, r));
break;
}
}
return ;
}

4105: [Thu Summer Camp 2015]平方运算的更多相关文章

  1. bzoj:4105: [Thu Summer Camp 2015]平方运算

    Description   Input 第一行有三个整数N,M,p,分别代表序列的长度.平方操作与询问操作的总次数以及在平方操作中所要模的数.   接下来一行N个数代表一开始的序列{X1,X2,... ...

  2. 2018.10.18 bzoj4105: [Thu Summer Camp 2015]平方运算(线段树)

    传送门 线段树妙题. 显然平方几次就会循环(打表证明不解释). 然后所有环长度的lcmlcmlcm不大于70. 因此维护一下当前区间中的节点是否全部在环上. 不是直接暴力到叶子节点修改. 否则整体打标 ...

  3. BZOJ4105 [Thu Summer Camp 2015]平方运算 【线段树】

    题目链接 BZOJ4105 题解 平方操作orz,虽说应该是线段树,但是不会维护啊QAQ 小瞧一眼题解... 平方成环?环长\(lcm\)小于\(60\)? 果然还是打表找规律题.... 那就很好做了 ...

  4. bzoj4105: [Thu Summer Camp 2015]平方运算

    填坑 我不知道怎么算的,但是所有环的LCM数不会超过60 然后用线段树维护这个东西,每个节点记录子树内的循环节 没到循环节的暴力枚举 复杂度是nlogn再乘以循环节长度 #include<cst ...

  5. [Thu Summer Camp 2015]解密运算

    4104: [Thu Summer Camp 2015]解密运算 Time Limit: 10 Sec  Memory Limit: 512 MB Description 对于一个长度为N的字符串,我 ...

  6. 【BZOJ 4104】 4104: [Thu Summer Camp 2015]解密运算 (智商)

    4104: [Thu Summer Camp 2015]解密运算 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 370  Solved: 237 De ...

  7. BZOJ4104:[Thu Summer Camp 2015]解密运算——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4104 对于一个长度为N的字符串,我们在字符串的末尾添加一个特殊的字符".".之 ...

  8. BZOJ4104 [Thu Summer Camp 2015]解密运算 【乱搞】

    题目链接 BZOJ4104 题解 我们将已知字符排序,由循环就可以得到一个对应关系 如样例就是: 0->第5行 1->第1行 1->第2行 1->第3行 1->第5行 2 ...

  9. bzoj 4104 [Thu Summer Camp 2015]解密运算——思路

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4104 想了很久,想出一个 nlogn (也许是 n2logn )的,可惜空间是 n2 . 已 ...

随机推荐

  1. 怎样简单的制作一个CHM格式的帮助文档?

    http://jingyan.baidu.com/article/d8072ac446eb7bec95cefd0e.html 怎么制作CHM格式电子书 http://jingyan.baidu.com ...

  2. Python中下划线---完全解读

    Python 用下划线作为变量前缀和后缀指定特殊变量 _xxx 不能用'from module import *'导入 __xxx__ 系统定义名字 __xxx 类中的私有变量名 核心风格:避免用下划 ...

  3. php 模式

    设计模式1.单例模式类的计划生育1.让该类在外界无法造对象2.让外界可以造一个对象,做一个静态方法返回对象3.在类里面通过静态变量控 class Dog { static $dx; public $t ...

  4. android 菜单的总结

    安卓菜单有三种菜单. 选项菜单: 点击系统菜单按钮会触发 上下文菜单:长按屏幕触发 子菜单:某一个菜单的下一级菜单 具体的描叙:http://blog.csdn.net/zqiang_55/artic ...

  5. Microsoft Visual 的变态

    Microsoft Visual 里面使用指针 的时候, 声明要放在函数开始的位置,否则报错,真变态啊 刚刚发现,C的变量必须在语块开始声明,后面声明会报错,太不灵活了

  6. bzoj3289

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=3289 题目大意:Mato同学从各路神犇以各种方式(你们懂的)收集了许多资料,这些资料一共有n ...

  7. 请使用支持 JDBC 4.0 的 sqljdbc4.jar 类库

    转载请使用支持 JDBC 4.0 的 sqljdbc4.jar 类库 1.下载最新的JDBC(2012/3/6) http://www.microsoft.com/downloads/zh-cn/de ...

  8. php 常用代码段

    1.写文件 $fp = fopen("jsapi_ticket.json", "w+"); fwrite($fp, $str); fclose($fp); 2. ...

  9. 贴一份用delphi修改注册表改网卡MAC地址的代码

    //提示:此代码需要use Registry, Common; function WriteMAC(model:integer):integer; var reg:TRegistry; begin r ...

  10. Delphi-IP地址的隐藏

    IP地址的隐藏 一.前言 本文主要介绍如何在程序中实现IP地址的隐藏.其实这篇东西不算我写的.其中<IP头结构>部分我懒得打字,故复制.粘贴了孤独剑客的文章,先说声谢谢!代码部分参考了外国 ...