CF633G
题目大意:
给你一棵树,根节点为1
有2种操作,第一种是给u节点所在的子树的所有节点的权值+x
第二种是询问,假设v是子树u中的节点,有多少种质数p满足av = p + m·k
做法:维护子树信息显然dfs序,考虑用线段树维护一个区间内有哪些值
每个区间用一个bitset维护
这里有一个小技巧,bitset都+x然后%m,可以bt2[k]=(bt2[k]<<y)|(bt2[k]>>(m-y));
然后求出区间的bitset,和质数的bitset&一下,求1的个数即可
代码如下:
#include<bits/stdc++.h>
#define N 500005
using namespace std;bool vis[N];
int n,m,Q,x,y,opt,tim1,kk,tot,head[N],a[N],dfn[N],sz[N],pr[N];
bitset<1005>bt1,bt2[N],bt3;int ee[N],reall[N];
struct Tree{int nxt,to;}e[N];
inline void link(int x,int y){e[++kk].nxt=head[x];e[kk].to=y;head[x]=kk;}
inline void init(){
for (int i=2;i<m;i++){
if (!vis[i]) pr[++tot]=i;
for (int j=1;j<=tot;j++){
if (i*pr[j]>=m) break;
vis[i*pr[j]]=true;
if (i%pr[j]==0) break;
}
}
}
void dfs(int u,int fa){
sz[u]=1;dfn[u]=++tim1;reall[tim1]=a[u];
for (int i=head[u];i;i=e[i].nxt){
int v=e[i].to;
if (v==fa) continue;
dfs(v,u);sz[u]+=sz[v];
}
}
void build(int k,int l,int r){
if (l==r){bt2[k][reall[l]]=1;return;}
int mid=(l+r)>>1;
build(k*2,l,mid);build(k*2+1,mid+1,r);
bt2[k]=bt2[k*2]|bt2[k*2+1];
}
void update(int k,int y){y%=m;bt2[k]=(bt2[k]<<y)|(bt2[k]>>(m-y));}
inline void pushdown(int k){
if (!ee[k]) return;
ee[k*2]=(ee[k*2]+ee[k])%m;
ee[k*2+1]=(ee[k*2+1]+ee[k])%m;
update(k*2,ee[k]);update(k*2+1,ee[k]);
ee[k]=0;
}
void add(int k,int l,int r,int x,int y,int z){
if (l!=r) pushdown(k);
if (x<=l&&y>=r){update(k,z);ee[k]=(ee[k]+z)%m;return;}
int mid=(l+r)>>1;
if (y<=mid) add(k*2,l,mid,x,y,z);
else if (x>mid) add(k*2+1,mid+1,r,x,y,z);
else add(k*2,l,mid,x,mid,z),add(k*2+1,mid+1,r,mid+1,y,z);
bt2[k]=bt2[k*2]|bt2[k*2+1];
}
void query(int k,int l,int r,int x,int y){
if (l!=r) pushdown(k);
if (x<=l&&y>=r){bt3=bt3|bt2[k];return;}
int mid=(l+r)>>1;
if (y<=mid) query(k*2,l,mid,x,y);
else if (x>mid) query(k*2+1,mid+1,r,x,y);
else query(k*2,l,mid,x,mid),query(k*2+1,mid+1,r,mid+1,y);
}
int main(){
scanf("%d%d",&n,&m);
for (int i=1;i<=n;i++) scanf("%d",&a[i]),a[i]%=m;
for (int i=1;i<n;i++){
scanf("%d%d",&x,&y);
link(x,y);link(y,x);
}
init();
for (int j=1;j<=tot;j++) bt1[pr[j]]=1;
//printf("%d\n",(int)bt1.count());
dfs(1,-1);
//for (int i=1;i<=n;i++) printf("%d ",dfn[i]);puts("");
//for (int i=1;i<=n;i++) printf("%d ",reall[i]);puts("");
build(1,1,n);
//if (bt2[7][5]) puts("Yes1");
/*if (bt2[3][5]) puts("Yes1");//
if (bt2[3][1]) puts("Yes2");
if (bt2[2][7]) puts("YES1");
if (bt2[2][8]) puts("YES2");
if (bt2[2][0]) puts("YES3");
if (bt2[1][8]) puts("Yes1");
if (bt2[1][7]) puts("Yes2");
if (bt2[1][5]) puts("Yes3");
if (bt2[1][1]) puts("Yes4");
if (bt2[1][0]) puts("Yes5");*/
scanf("%d",&Q);
while (Q--){
scanf("%d%d",&opt,&x);
if (opt==1){
scanf("%d",&y);y%=m;
if (y) add(1,1,n,dfn[x],dfn[x]+sz[x]-1,y);
}
else {
/*if (bt2[1][0]) puts("Yes1");
if (bt2[1][9]) puts("Yes2");
if (bt2[1][7]) puts("Yes3");
if (bt2[1][3]) puts("Yes4");
if (bt2[1][2]) puts("Yes5");*/
bt3.reset();
query(1,1,n,dfn[x],dfn[x]+sz[x]-1);
//printf("%d\n",(int)bt3.count());
bt3=bt3&bt1;
printf("%d\n",(int)bt3.count());
}
}
return 0;
}
CF633G的更多相关文章
随机推荐
- (NO.00004)iOS实现打砖块游戏(七):关卡类的实现
大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请告诉我,如果觉得不错请多多支持点赞.谢谢! hopy ;) 关卡游戏的精髓都集中在游戏的关卡里,其中包含了游戏的所有要素,至 ...
- Linux_Oracle命令大全
一,启动 1.#su - oracle 切换到oracle用户且切换到它的环境 2.$lsnrctl status 查看监听及数据库状态 3.$ls ...
- 如何在Cocos2D游戏中实现A*寻路算法(三)
大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请告诉我,如果觉得不错请多多支持点赞.谢谢! hopy ;) 免责申明:本博客提供的所有翻译文章原稿均来自互联网,仅供学习交流 ...
- 《java入门第一季》之有趣的集合小案例---获取10个【1-20之间】的随机数,要求不能重复。
import java.util.ArrayList; import java.util.Random; /* * 获取10个[1-20之间]的随机数,要求不能重复.(注意:不是获取10个数,如果单纯 ...
- Gradle 笔记——Java构建入门
Gradle是一个通用的构建工具,通过它的构建脚本你可以构建任何你想要实现的东西,不过前提是你需要先写好构建脚本的代码.而大部分的项目,它们的构建流程基本是一样的,我们不必为每一个工程都编写它的构建代 ...
- Swift基础之PickerView(时间)选择器
代码讲解:(后面有额外代码讲解) 首页设计UIPickerView的样式设计: leftArray = ["花朵","颜色","形状"]; ...
- 分布式进阶(十二)Docker固定Container IP
使用pipework工具. 前提:每个Container所做的工作现在还很少,可以不用save.commit. 为了便于通信,自定义一个网桥(192.168.1.180/24),使之IP与宿主主机IP ...
- Java实现栈之计算器
Java实现栈来做一个将中缀表达式转化为后缀表达式的程序,中缀表达式更符合我们的主观感受,后缀表达式更适合计算机的运算,下面直接上代码吧: package Character1; import jav ...
- 解决log4cxx退出时的异常
解决log4cxx退出时的异常(金庆的专栏)如果使用log4cxx的FileWatchdog线程来监视日志配置文件进行动态配置,就可能碰到程序退出时产生的异常.程序退出时清理工作耗时很长时,该异常很容 ...
- java 二进制数字符串转换工具类
java 二进制数字符串转换工具类 将二进制转换成八进制 将二进制转换成十进制 将二进制转换成十六进制 将十进制转换成二进制 package com.iteye.injavawetrust.ad; i ...