XJOI——NOIP2015提高组模拟题19-day1——观光旅行

时间:2023-03-09 01:40:10
XJOI——NOIP2015提高组模拟题19-day1——观光旅行

http://www.hzxjhs.com:83/contest/493/problem/3

【题目大意】

给定一个有n(n<=500000)个点,m(1<=500000)条边的无向图。给Q(1<=500000)个询问ui和vi,问ui和vi之间是否存在一条不经过重复点的路径,使得经过的点数为偶数。

【题目解析】

XJOI——NOIP2015提高组模拟题19-day1——观光旅行

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cctype> using namespace std; #define re(i,a,b) for(i=(a);i<=(b);i++)
#define red(i,a,b) for(i=(a);i>=(b);i--) #define SF scanf
#define PF printf
#define mmst(a,v) memset(a,v,sizeof(a)) int gint()
{
int res=;bool neg=;char z;
for(z=getchar();z!=EOF && z!='-' && !isdigit(z);z=getchar());
if(z==EOF)return ;
if(z=='-'){neg=;z=getchar();}
for(;z!=EOF && isdigit(z);res=res*+z-'',z=getchar());
return (neg)?-res:res;
} const int maxn=; int n,m,Q;
int now,info[maxn+];
struct Tedge{int v,next;}edge[*maxn+];
int treeedge[*maxn+]; void addedge(int u,int v){now++;edge[now].v=v;edge[now].next=info[u];info[u]=now;} int dep[maxn+],jump[maxn+][];
int odd[maxn+],f[maxn+]; void dfs(int u)
{
int i,j,v;
for(i=info[u],v=edge[i].v;i!=-;i=edge[i].next,v=edge[i].v)
if(!dep[v])
{
dep[v]=dep[u]+;
jump[v][]=u;
re(j,,)jump[v][j]=jump[jump[v][j-]][j-];
treeedge[i]=treeedge[i^]=;
dfs(v);
odd[u]+=odd[v];
}
else
if(!treeedge[i] && dep[u]>dep[v] && (dep[u]-dep[v])%==)
{
odd[u]++;
odd[v]--;
}
} void dfs2(int u)
{
int i,v;
f[u]=f[jump[u][]]+odd[u];
for(i=info[u],v=edge[i].v;i!=-;i=edge[i].next,v=edge[i].v)if(treeedge[i] && dep[v]>dep[u])dfs2(v);
} void swim(int &x,int H){int i;for(i=;H!=;H>>=,i++)if(H&)x=jump[x][i];}
int ask_lca(int x,int y)
{
if(dep[x]<dep[y])swap(x,y);
swim(x,dep[x]-dep[y]);
if(x==y)return x;
int j;
red(j,,)if(jump[x][j]!=jump[y][j])x=jump[x][j],y=jump[y][j];
return jump[x][];
} int main()
{
freopen("travel.in","r",stdin);
freopen("travel.out","w",stdout);
int i,j;
n=gint();m=gint();Q=gint();
now=-;mmst(info,-);
re(i,,m)
{
int x=gint(),y=gint();
addedge(x,y);addedge(y,x);
}
dep[]=;
re(j,,)jump[][j]=;
dfs();
re(i,,n)if(odd[i])odd[i]=;
dfs2();
while(Q--)
{
int x=gint(),y=gint();
int lca=ask_lca(x,y);
if((dep[x]-dep[lca]+dep[y]-dep[lca])%== || f[x]-f[lca]>= || f[y]-f[lca]>=)PF("TAK\n");else PF("NIE\n");
}
return ;
}