BZOJ 1316: 树上的询问 (点分治+set)
2015-11-22 08:57
302 查看
题目:http://www.lydsy.com/JudgeOnline/problem.php?id=1316
因为只要求存在某条路径长度为K,所以点分,然后用set判断差值是否在set中就可以了。
#include<cstring>
#include<algorithm>
#include<iostream>
#include<cstdio>
#include<queue>
#include<set>
#include<cmath>
#define rep(i,l,r) for (int i=l;i<=r;i++)
#define down(i,l,r) for (int i=l;i>=r;i--)
#define clr(x,y) memset(x,y,sizeof(x))
#define maxn 10900
#define inf int(1e9)
#define mm 1000000007
#define esp 1e-6
using namespace std;
#define ll long long
struct data{int obj,pre; ll c;
}e[maxn*2];
int head[maxn],s[maxn],ans[maxn],q[maxn],vis[maxn];
int n,m,tot,sum,mn,rt;
set<ll> bst;
int read(){
int x=0,f=1; char ch=getchar();
while (!isdigit(ch)) {if (ch=='-') f=-1; ch=getchar();}
while (isdigit(ch)) x=x*10+ch-'0',ch=getchar();
return x*f;
}
void insert(int x,int y,int z){
e[++tot].obj=y; e[tot].pre=head[x]; e[tot].c=z; head[x]=tot;
}
void dfs(int u,int fa){
s[u]=1;
int mx=0;
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (!vis[v]&&v!=fa){
dfs(v,u);
s[u]+=s[v];
mx=max(mx,s[v]);
}
}
mx=max(sum-mx,mx);
if (mx<mn) mn=mx,rt=u;
}
void add(int u,int fa,ll w){
bst.insert(w);
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (v!=fa&&!vis[v])add(v,u,w+e[j].c);
}
}
void update(int u,int fa,ll w){
rep(i,1,m) if (!ans[i])
ans[i]|=(bst.find(q[i]-w)!=bst.end());
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (v!=fa&&!vis[v]) update(v,u,w+e[j].c);
}
}
void solve(int u){
mn=inf;
dfs(u,0);
u=rt;
bst.clear(); bst.insert(0);
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (!vis[v]) update(v,u,e[j].c),add(v,u,e[j].c); //先判断点再加边。
}
vis[u]=1;
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (!vis[v]){
sum=s[v];
solve(v);
}
}
}
int main(){
n=read(); m=read();
int x,y,z;
rep(i,1,n-1){
x=read(); y=read(); z=read();
insert(x,y,z);
insert(y,x,z);
}
rep(i,1,m) q[i]=read();
sum=n;
solve(1);
rep(i,1,m) if(!q[i]) ans[i]=1; //如果q[i]=0,那么可以只选取一个点。
rep(i,1,m) if (ans[i]) puts("Yes"); else puts("No");
return 0;
}
因为只要求存在某条路径长度为K,所以点分,然后用set判断差值是否在set中就可以了。
#include<cstring>
#include<algorithm>
#include<iostream>
#include<cstdio>
#include<queue>
#include<set>
#include<cmath>
#define rep(i,l,r) for (int i=l;i<=r;i++)
#define down(i,l,r) for (int i=l;i>=r;i--)
#define clr(x,y) memset(x,y,sizeof(x))
#define maxn 10900
#define inf int(1e9)
#define mm 1000000007
#define esp 1e-6
using namespace std;
#define ll long long
struct data{int obj,pre; ll c;
}e[maxn*2];
int head[maxn],s[maxn],ans[maxn],q[maxn],vis[maxn];
int n,m,tot,sum,mn,rt;
set<ll> bst;
int read(){
int x=0,f=1; char ch=getchar();
while (!isdigit(ch)) {if (ch=='-') f=-1; ch=getchar();}
while (isdigit(ch)) x=x*10+ch-'0',ch=getchar();
return x*f;
}
void insert(int x,int y,int z){
e[++tot].obj=y; e[tot].pre=head[x]; e[tot].c=z; head[x]=tot;
}
void dfs(int u,int fa){
s[u]=1;
int mx=0;
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (!vis[v]&&v!=fa){
dfs(v,u);
s[u]+=s[v];
mx=max(mx,s[v]);
}
}
mx=max(sum-mx,mx);
if (mx<mn) mn=mx,rt=u;
}
void add(int u,int fa,ll w){
bst.insert(w);
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (v!=fa&&!vis[v])add(v,u,w+e[j].c);
}
}
void update(int u,int fa,ll w){
rep(i,1,m) if (!ans[i])
ans[i]|=(bst.find(q[i]-w)!=bst.end());
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (v!=fa&&!vis[v]) update(v,u,w+e[j].c);
}
}
void solve(int u){
mn=inf;
dfs(u,0);
u=rt;
bst.clear(); bst.insert(0);
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (!vis[v]) update(v,u,e[j].c),add(v,u,e[j].c); //先判断点再加边。
}
vis[u]=1;
for (int j=head[u];j;j=e[j].pre){
int v=e[j].obj;
if (!vis[v]){
sum=s[v];
solve(v);
}
}
}
int main(){
n=read(); m=read();
int x,y,z;
rep(i,1,n-1){
x=read(); y=read(); z=read();
insert(x,y,z);
insert(y,x,z);
}
rep(i,1,m) q[i]=read();
sum=n;
solve(1);
rep(i,1,m) if(!q[i]) ans[i]=1; //如果q[i]=0,那么可以只选取一个点。
rep(i,1,m) if (ans[i]) puts("Yes"); else puts("No");
return 0;
}
相关文章推荐
- 黑客丛林之旅 第十关
- 关于矩阵重复元素检测判断条件的粗陋看法
- linux下tar命令参数详解
- Mac终端编译运行C++
- matlab:cell的元素的调用—— acell(2)相比acell{2}
- C#调用java类、jar包方法
- Sailfish OS 开发商 Jolla 融资失败,裁员过半
- ASP.NET MVC- ActionFilter的使用
- JS实现别踩白块小游戏
- GoogleDoc - 温故而知新Activity生命周期方法
- 推荐个好用的安卓记事本便签软件
- matlab:把cell中的某个元素删去
- java API中Util包
- JQuery零碎知识点归纳
- 单向链表 (Singly linked list) 的C++实现
- github android优秀的开源项目、作者推荐
- HDR(High Dynamic Range Imaging)在摄影中指高动态范围成像
- 26,字符串函数
- 成为Java高手的25个学习要点
- 检测IE