首先嘛这道题目只要知道一个东西就很容易了:所有循环的最小公约数<=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. 如何获得keyhash

    keyhash是由keystore产生的 三个命令解决问题 keytool -exportcert -alias androiddebugkey -keystore debug.keystore &g ...

  2. JV的DOM操作

    一.基本概念 :是文档对象模型,这种模型为树模型:文档指标签文档:对象是指文档中每个元素:模型是指抽象化的东西. :.Windows对象操作:.属性:opener:(打开当前窗口的原窗口.)dialo ...

  3. console 高级用法

    console.log()用法,相信大家都很熟悉了,这里就不再啰嗦. 下面来玩几个新鲜点的,我用的是chrome28,不保证兼容其他浏览器:console.log的第一个参数中可以指定一个格式字符,这 ...

  4. Objective-C Effective 技巧

    1.除非有必要,否则不要引用头文件,一般来说应该利用@class使用前向声明,并在实现中引用头文件:如果实在无法使用,比如要声明某个类遵循一项协议,这种情况下,尽量把这条声明移到分类中,如果不行的话, ...

  5. 每一个Servlet只有一个实例,多个线程

    每一个Servlet只有一个实例,多个线程: Servlet: package com.stono.servlet.synchronize; import javax.servlet.http.Htt ...

  6. Request中的方法调用

    几种方法的调用: package com.stono.servlet; import java.io.IOException; import java.util.Enumeration; import ...

  7. 统计SQLSERVER表行数,以及每天数据变化的行数

    此sql对监控系统很有帮助,知道哪些表压力大,每天的数量级大概多少等信息. 得到这些信息就可以做相应的策略来进行系统优化. create table tmp( name varchar(50), ro ...

  8. 清理微信浏览网站的缓存,Cookie

    微信官方说明是取消关注,但是开发中发现取消关注缓存还是存在! 解决方法如下: 方法一: 用微信内置浏览器打开这个网页debugx5.qq.com ,就会有清除缓存的选项,如下图 方法二: 如果你用An ...

  9. HDU5477(模拟)

    A Sweet Journey Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)T ...

  10. 作为测试人员,我是这么报BUG的

    在测试人员提需求的时候,大家经常会看到,测试员和开发一言不合就上BUG.然后开发一下就炸了,屡试不爽,招招致命. 曾经看到有个段子这么写道: 不要对程序员说,你的代码有BUG. 他的第一反应是:1.你 ...