首先嘛这道题目只要知道一个东西就很容易了:所有循环的最小公约数<=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. Memcached源码分析之slabs.c

    #include "memcached.h" #include <sys/stat.h> #include <sys/socket.h> #include ...

  2. 快速搭建LNMP

    [centos 6.4 server]系统安装请参考:http://blog.zhuyin.org/748.html1.防火墙设置: iptables -F service iptables save ...

  3. HTTP协议快速入门

    一.定义 The Hypertext Transfer Protocol (HTTP) is an application protocol for distributed, collaborativ ...

  4. java求两个集合的差集

    public static void main(String[] args) {Set set = new HashSet();Set set1 = new HashSet();set.add(&qu ...

  5. 获取IE浏览器关闭事件

    //关闭浏览器时才会触发此操作,刷新页面不执行 //n 检测鼠标相对于用户屏幕的水平位置 - 网页正文部分左:求出鼠标在当前窗口上的水平位置(参照:当前窗口右上角为0.0坐标) //m 网页正文全文宽 ...

  6. 一分钟完成MySQL5.7安装部署

    原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://suifu.blog.51cto.com/9167728/1855415 Part ...

  7. Cell.reuseIdentifier 指什么

    Cell.reuseIdentifier 指的是 默认为空,如果不定义,在执行 [_tableView registerNib:templateCellNib forCellReuseIdentifi ...

  8. PHP文件上传主要代码讲解

    导读:在php开发过程中,文件上传也经常用到,这里简单介绍下. 在php开发过程中,文件上传也经常用到,这里简单介绍下. 代码如下: <?php    if($_FILES['myfile'][ ...

  9. redis 高级配置

    一.安全性 设置密码:在配置文件中设置 requirepass 123456 由于redis的速度非常快,每秒可以进行15万次的暴力破解,所以密码设置要强壮些 在客户端登录或者连接的时候,使用 aut ...

  10. 在官网下载了最新版的PHP,解压后的安装包里为什么没有php5isapi.dll这个dll文件?

    因为自PHP 5.3.1版本开始,PHP便已不在支持ISAPI模式,所以你在PHP5.3.1版本以上的php目录中看不到php5isapi.dll文件. 那么,IIS6下跑PHP 5.3.1以上版本时 ...