题目链接: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. Asp.Net Core Authentication Middleware And Generate Token

    .mytitle { background: #2B6695; color: white; font-family: "微软雅黑", "宋体", "黑 ...

  2. hdoj 1175 (bfs)

    题意: 判断两点之间是否可以通过至多有两次转变方向以达到相连,就是平时玩的连连看游戏,但是不能从外面绕过去. 思路:bfs,给每个加入的队列的点添加转变方向次数turn和点当前要走的方向dir属性,起 ...

  3. Leetcode_001_TwoSum_求和为固定数的两个数的索引

    题目描述    给定一个整型数组,在数组中找出两个数使这两个数的和为给定数,从小到大输出这两个数在数组中的位置(我们可以假定输出结果只有一个).例如,输入:N={1,4,8,20}, target=1 ...

  4. 最短路径之Dijkstras算法(图片格式)

  5. 梳理一下web总的一些概念

    servlet中的类适合繁复翻看文档,熟悉各个类的常用方法,看一些经典的案例代码. ServletConfig 每个项目有多个servlet,每个servlet对应一个ServletCOnfigt对象 ...

  6. 性能测试平台效率优化的一次经验(python版)

    在做性能测试平台的优化过程中,由于启动任务相对其他测试任务比较频繁,而目前30次两个包的交叉对比(30次)测试需要耗时30分钟整,因此打算优先对测试流程做一次优化,将测试时间消耗降低到20分钟. 由于 ...

  7. flex中日期的格式化

    今天我做的项目中需要把时间给拆分了,形式为:yyyy-MM-DD HH mm, 下面是我的代码实现: <?xml version="1.0" encoding="u ...

  8. python运用中文注释时报错解决方法

    写了一段简单的代码,不知 为什么总是报错,后来上网查了一下才知道原因,当用中文进行注释时需要添加如下代码:# coding=utf-8          (注意:该段代码必须放在最前面才能有用,并且 ...

  9. 深入理解ajax系列第九篇——jQuery中的ajax

    前面的话 jQuery提供了一些日常开发中需要的快捷操作,例如load.ajax.get和post等,使用jQuery开发ajax将变得极其简单.这样开发人员就可以将程序开发集中在业务和用户体验上,而 ...

  10. Alamofire源码解读系列(十)之序列化(ResponseSerialization)

    本篇主要讲解Alamofire中如何把服务器返回的数据序列化 前言 和前边的文章不同, 在这一篇中,我想从程序的设计层次上解读ResponseSerialization这个文件.更直观的去探讨该功能是 ...