Chosen Problem Solving and Program design as an optional course, you are required to solve all kinds of problems. Here, we get a new problem.

There is a very long board with length L centimeter, L is a positive integer, so we can evenly divide the board into L segments, and they are labeled by 1, 2, ... L from left to right, each is 1 centimeter long. Now we have to color the board - one segment with only one color. We can do following two operations on the board:

1. "C A B C" Color the board from segment A to segment B with color C. 
2. "P A B" Output the number of different colors painted between segment A and segment B (including).

In our daily life, we have very few words to describe a color (red, green, blue, yellow…), so you may assume that the total number of different colors T is very small. To make it simple, we express the names of colors as color 1, color 2, ... color T. At the beginning, the board was painted in color 1. Now the rest of problem is left to your.

Input

First line of input contains L (1 <= L <= 100000), T (1 <= T <= 30) and O (1 <= O <= 100000). Here O denotes the number of operations. Following O lines, each contains "C A B C" or "P A B" (here A, B, C are integers, and A may be larger than B) as an operation defined previously.

Output

Ouput results of the output operation in order, each line contains a number.

Sample Input

2 2 4
C 1 1 2
P 1 2
C 2 2 2
P 1 2

Sample Output

2
1
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<vector>
#include<string>
#include<cstring>
using namespace std;
typedef long long LL;
const int MAXN = 1e7 + ;
const int N = 3e5 + ;
LL euler[MAXN];
void geteuler()
{
memset(euler, , sizeof(euler));
euler[] = ;
for (LL i = ; i < MAXN; i++)
{
if (!euler[i])
for (LL j = i; j < MAXN; j += i)
{
if (!euler[j]) euler[j] = j;
euler[j] = euler[j] / i * (i - );
}
}
}
struct node
{
int l, r;
LL sum, laz;
}T[N * + ];
LL a[N];
void pushup(int p)
{
T[p].sum = T[p * ].sum + T[p * + ].sum;
if (T[p * ].laz == T[p * + ].laz)
T[p].laz = T[p * ].laz;
else
T[p].laz = ;
}
void pushdown(int p)
{
if (T[p].laz)
{
T[p * ].laz = T[p * + ].laz = T[p].laz;
T[p * ].sum = T[p].laz * (T[p * ].r - T[p * ].l + );
T[p * + ].sum = T[p].laz * (T[p * + ].r - T[p * + ].l + );
}
}
void update1(int x, int l, int r)
{
if (T[x].laz&&T[x].l == l&&T[x].r == r)
{
T[x].laz = euler[T[x].laz];
T[x].sum = T[x].laz * (T[x].r - T[x].l + );
return;
}
pushdown(x);
int mid = (T[x].l + T[x].r) / ;
if (r <= mid)
update1(x * , l, r);
else if(l > mid)
update1(x * + , l , r);
else
{
update1(x * , l, mid);
update1(x * + , mid + , r);
}
pushup(x);
}
void update2(int x, int l, int r, LL val)
{
if (l == T[x].l&&r == T[x].r)
{
T[x].laz = val;
T[x].sum = (T[x].r - T[x].l + )*T[x].laz;
return;
}
pushdown(x);
int mid = (T[x].l + T[x].r) / ;
if (r <= mid)
update2(x * , l, r, val);
else if (l > mid)
update2(x * + , l, r, val);
else
{
update2(x * , l, mid, val);
update2(x * + , mid + , r, val);
}
pushup(x);
} void build(int x, int l, int r)
{
T[x].l = l, T[x].r = r;
T[x].laz = T[x].sum = ;
if (l == r)
{
T[x].laz = T[x].sum = a[l];
return;
}
int mid = (l + r) / ;
build(x * , l, mid);
build(x * + , mid + , r);
pushup(x);
} LL query(int x, int l, int r)
{
if (T[x].l == l&&T[x].r == r)
return T[x].sum;
int mid = (T[x].l + T[x].r) / ;
pushdown(x);
if (r <= mid)
return query(x * , l, r);
else if (l > mid)
return query(x * + , l, r);
else
return query(x * , l, mid) + query(x * + , mid + , r);
}
int t, n, m;
int main()
{
geteuler();
ios::sync_with_stdio();
scanf("%d", &t);
while (t--)
{
scanf("%d%d", &n, &m);
for (int i = ; i <= n; i++)
scanf("%lld", &a[i]);
build(, , n);
int op, L, R;
LL tmp;
while (m--)
{
scanf("%d%d%d", &op, &L, &R);
if (op == )
{
update1(, L, R);
}
else if (op == )
{
scanf("%lld", &tmp);
update2(, L, R, tmp);
}
else if (op == )
{
printf("%lld\n", query(, L, R));
}
}
}
}

Count Color

 
修改节点的值,查询区间总和
这里laz就表示当前区间元素是否相同 pushdown 顺推
pushup
只有当左右两边都是整块而且左右边的颜色相等才能设置laz =
这里多了一个左右都是整块的条件是因为在欧拉的题目中laz>0就表示是整块了 Rikka with Phi 
laz有两个含义:laz== 表示当前区间多个元素值不同
laz == x 表示当前区间元素的值都是x
修改节点的值,查询区间总和
pushdown
顺推即可,laz 相同, sum计算一下
pushup
当前sum = 子区间sum之和
当前laz = 子区间laz 相同? 子区间laz,否则为0 分为块状区域统一处理,当处理比当前块更小的块的时候,把之前积累的信息传递下去,递归处理 两个题的区别在于颜色的题目不需要Laz来表示当前值,当前值用color表示即可 PUSHDOWN
更新结点数据
PUSHUP
根据结点更新当前点的数据

最新文章

  1. Demo源码放送:打通B/S与C/S !让HTML5 WebSocket与.NET Socket公用同一个服务端!
  2. vi学习总结
  3. C# Mvc异常处理过滤器
  4. 命名规范(数据库,c#)
  5. grep 命令详解
  6. java内存模型-final
  7. Java反射机制&lt;1&gt;
  8. php中rsa加密解密验证
  9. VC5509的通用GEL代码
  10. git 创建一个新分支,并将一个分支内容复制给创建的新分支
  11. java常用linux命令
  12. motan源码分析四:客户端调用服务
  13. 简单的php和apache的安装
  14. C - 哗啦啦村的扩建
  15. Storyboard中拖拽控件不能运行的问题(在运行的时候,相应的控件代码没有被执行)
  16. 图文详解如何快捷搭建LNMP服务环境
  17. ORA-00922:选项缺失或无效
  18. Linux iptables用法与NAT
  19. MyBatis insert/delete/update 的返回值
  20. 肺结节CT影像特征提取(四)——肺结节CT影像特征提取MATLAB代码实现

热门文章

  1. 如何看Spring源码
  2. R in action读书笔记(12)第九章 方差分析
  3. spring 获取ApplicationContext
  4. bat 时间 的运算与提取
  5. 四则运算 来自 http://www.cnblogs.com/ys1101/p/4368103.html
  6. springMvc(初识+操作步骤)
  7. 火狐删除配置文件 会删除目录下所有文件 切记不要把配置文件建立在桌面 恢复软件:易我数据恢复向导 9.0 DiskGenius500
  8. java网络编程_IP地址
  9. Learning Discriminative and Transformation Covariant Local Feature Detectors实验环境搭建详细过程
  10. 诊断:RHEL7安装11.2RAC时root.sh错误ohasd failed to start