题目链接:http://codeforces.com/problemset/problem/633/G

大意是一棵树两种操作,第一种是某一节点子树所有值+v,第二种问子树中节点模m出现了多少种m以内的质数。

第一种操作非常熟悉了,把每个节点dfs过程中的pre和post做出来,对序列做线段树。维护取模也不是问题。第二种操作,可以利用bitset记录质数出现情况。所以整个线段树需要维护bitset的信息。

对于某一个bitset x,如果子树所有值需要加y,则x=(x<<y)|(x>>(m-y))

一开始写挂了几次,有一点没注意到,因为我bitset直接全都是1000,而不是m,所以上面式子左移会有问题,解决方法是做一个0到m每位都是1的全集,或者表示质数集的bitset上限做到m。

 #include <iostream>
#include <vector>
#include <algorithm>
#include <string>
#include <string.h>
#include <stdio.h>
#include <math.h>
#include <stdlib.h>
#include <queue>
#include <stack>
#include <map>
#include <set>
#include <ctime>
#include <numeric>
#include <bitset>
#include <cassert> using namespace std;
const int N=;
int a[N];
vector<int>edge[N];
bitset<>bs[N<<];
bitset<>prime;
int mark[N<<];
int L[N],R[N];
int dfn=;
int m;
int id[N];
void dfs(int u,int f) {
L[u]=++dfn;
id[dfn]=u;
for (int i=;i<edge[u].size();i++) {
int v=edge[u][i];
if (v==f)
continue;
dfs(v,u);
}
R[u]=dfn;
}
void up(int rt) {
bs[rt]=(bs[rt<<]|bs[rt<<|]);
}
void add(int rt,int x) {
bs[rt]=(((bs[rt]<<x))|(bs[rt]>>(m-x)));
}
void down(int rt) {
if (mark[rt]) {
add(rt<<,mark[rt]);
add(rt<<|,mark[rt]);
mark[rt<<]=(mark[rt<<]+mark[rt])%m;
mark[rt<<|]=(mark[rt<<|]+mark[rt])%m;
mark[rt]=;
}
}
void build(int l,int r,int rt) {
mark[rt]=;
if (l==r) {
int x=a[id[l]];
bs[rt].set(x);
return;
}
int m=(l+r)>>;
build(l,m,rt<<);
build(m+,r,rt<<|);
up(rt);
} void upd(int L,int R,int x,int l,int r,int rt) {
if (L<=l&&r<=R) {
add(rt,x);
mark[rt]=(mark[rt]+x)%m;
return;
}
down(rt);
int m=(l+r)>>;
if (L<=m)
upd(L,R,x,l,m,rt<<);
if (R>m)
upd(L,R,x,m+,r,rt<<|);
up(rt);
}
bitset<> ans;
void ask(int L,int R,int l,int r,int rt) {
if (L<=l&&r<=R) {
ans|=bs[rt];
return;
}
down(rt);
int m=(l+r)>>;
if (L<=m)
ask(L,R,l,m,rt<<);
if (R>m)
ask(L,R,m+,r,rt<<|);
} int main () {
int n;
scanf("%d %d",&n,&m);
for (int i=;i<m;i++) {
bool isp=true;
for (int j=;isp&&j*j<=i;j++) {
if (i%j==)
isp=false;
}
if (isp){
prime.set(i);
}
}
for (int i=;i<=n;i++){
scanf("%d",a+i);
a[i]%=m;
}
for (int i=;i<n;i++) {
int u,v;
scanf("%d %d",&u,&v);
edge[u].push_back(v);
edge[v].push_back(u);
}
dfn=;
dfs(,-);
build(,n,);
int Q;
scanf("%d",&Q);
while (Q--) {
int op;
scanf("%d",&op);
if (op==) {
int u,x;
scanf("%d %d",&u,&x);
x%=m;
upd(L[u],R[u],x,,n,);
}
else {
int u;
scanf("%d",&u);
ans=;
ask(L[u],R[u],,n,);
ans&=prime;
int ret=ans.count();
printf("%d\n",ret);
}
}
return ;
}

CF Manthan, Codefest 16 G. Yash And Trees 线段树+bitset的更多相关文章

  1. Manthan, Codefest 16 G. Yash And Trees dfs序+线段树+bitset

    G. Yash And Trees 题目连接: http://www.codeforces.com/contest/633/problem/G Description Yash loves playi ...

  2. G. Yash And Trees 线段树 + dfs序 + bitset

    这个是要用bitset 一个大整数的二进制 学习推荐博客 这个题目大意就是:给你n,m 还有一个序列,还有一个一棵树,有一种操作一种询问 操作是给你一个节点 把这个节点及其子节点都加上x 询问是 给你 ...

  3. CF #Manthan, Codefest 16 C. Spy Syndrome 2 Trie

    题目链接:http://codeforces.com/problemset/problem/633/C 大意就是给个字典和一个字符串,求一个用字典中的单词恰好构成字符串的匹配. 比赛的时候是用AC自动 ...

  4. CF Manthan, Codefest 16 B. A Trivial Problem

    数学技巧真有趣,看出规律就很简单了 wa 题意:给出数k  输出所有阶乘尾数有k个0的数 这题来来回回看了两三遍, 想的方法总觉得会T 后来想想  阶乘 emmm  1*2*3*4*5*6*7*8*9 ...

  5. Manthan, Codefest 16 C. Spy Syndrome 2 字典树 + dp

    C. Spy Syndrome 2 题目连接: http://www.codeforces.com/contest/633/problem/C Description After observing ...

  6. Manthan, Codefest 16

    暴力 A - Ebony and Ivory import java.util.*; import java.io.*; public class Main { public static void ...

  7. codeforces 633G. Yash And Trees dfs序+线段树+bitset

    题目链接 G. Yash And Trees time limit per test 4 seconds memory limit per test 512 megabytes input stand ...

  8. Manthan, Codefest 16 H. Fibonacci-ish II 大力出奇迹 莫队 线段树 矩阵

    H. Fibonacci-ish II 题目连接: http://codeforces.com/contest/633/problem/H Description Yash is finally ti ...

  9. DFS序+线段树+bitset CF 620E New Year Tree(圣诞树)

    题目链接 题意: 一棵以1为根的树,树上每个节点有颜色标记(<=60),有两种操作: 1. 可以把某个节点的子树的节点(包括本身)都改成某种颜色 2. 查询某个节点的子树上(包括本身)有多少个不 ...

随机推荐

  1. 【2017-2-17】VS基本应用及C#基础第一节(定义变量、输入及输出)

    一VS基本应用 (一)新建项目 新建项目可有多种方法例如: 1.  在VS起始页面建立新项目 2.  在集成环境中,通过"文件"/"新建"/"项目&q ...

  2. ASP日期格式化函数

    Public Function GetFormatDate(DateAndTime, para)On Error Resume NextDim y, m, d, h, mi, s, strDateTi ...

  3. 1000: A+B Problem(NetWork Flow)

    1000: A+B Problem Time Limit: 1 Sec  Memory Limit: 5 MBSubmit: 11814  Solved: 7318[Submit][Status][D ...

  4. DOM的内部插入和外部插入

    /*****************返回值都是最前面调用这个方法的元素 ***************** */ //这是外部插入 $("div").insertBefore($( ...

  5. git 关键词扩展

    用习惯了svn 的 关键词扩展.自然在git上也会寻找. 不知道啥是svn的关键词扩展? 简单列举下: 对于subversion来说,一些版本信息是不需要手动添加的,subversion提供了关键字替 ...

  6. 最短路径之BF算法+线性规划(图片格式)

  7. SQL Server中的Merge关键字 更新表数据

    简介 Merge关键字是一个神奇的DML关键字.它在SQL Server 2008被引入,它能将Insert,Update,Delete简单的并为一句.MSDN对于Merge的解释非常的短小精悍:”根 ...

  8. C++枚举类型详解

    原创作品,转载请注明来源:http://www.cnblogs.com/shrimp-can/p/5171110.html  一.枚举类型的定义 enum 类型名 {枚举值表}: 类型名是变量名,指定 ...

  9. Java程序员应当知道的10个面向对象设计原则

    面向对象设计原则是OOPS编程的核心, 但我见过的大多数Java程序员热心于像Singleton (单例) . Decorator(装饰器).Observer(观察者) 等设计模式,而没有把足够多的注 ...

  10. MCMC(四)Gibbs采样

    MCMC(一)蒙特卡罗方法 MCMC(二)马尔科夫链 MCMC(三)MCMC采样和M-H采样 MCMC(四)Gibbs采样 在MCMC(三)MCMC采样和M-H采样中,我们讲到了M-H采样已经可以很好 ...