zoukankan      html  css  js  c++  java
  • 部落冲突

    部落冲突https://www.luogu.org/problemnew/show/P3950

    题目背景

    在一个叫做(Travian)的世界里,生活着各个大大小小的部落。其中最为强大的是罗马、高卢和日耳曼。他们之间为了争夺资源和土地,进行了无数次的战斗。期间诞生了众多家喻户晓的英雄人物,也留下了许多可歌可泣的动人故事。
    其中,在大大小小的部落之间,会有一些道路相连,这些道路是(Travian)世界里的重要枢纽,简单起见,你可以把这些部落与部落之间相连的道路看作一颗树,可见每条道路对于(Travian)世界的重要程度。有了这些道路,建筑工人就可以通过这些道路进行友好外交啦。
    然而,事情并不会像想象的那样美好,由于资源的匮乏,相邻的部落(由一条道路相连的部落)之间经常会发生大大小小的冲突事件,更有甚者,会升级为部落之间的大型战争。
    为了避免误伤,每当两个相邻的部落之间发生大型战争之时,这两个部落间的道路是不允许通行的,对于一些强大的部落,甚至能与多个相邻的部落同时开战,同样的,这些战争地带的道路十分危险,是不可通行的。
    天下之势,分久必合,当两个部落经历了不打不相识的苦战之后,他们可以签订停战协议(暂时停战,以后依旧可能再次开战),这样,两个部落之间的道路又会重新恢复为可通行状态,建筑工人们又可以经过此地购买最新的大本营设计图纸来强大自己的部落了。
    为了简单起见,我们把各大战争事件按发起的时间顺序依次编号(最先发起的战争编号就为 (1),第二次战争编号就为 (2),以此类推),当两个部落停战之时,则会直接告诉你这场战争的编号,然后这场战争就载入了史册,不复存在了,当然,这并不会影响到其他战争的编号。
    建筑工人十分讨厌战争,因为战争,想从一个部落到另一个部落进行友好交流的建筑工人可能就此白跑一趟。所以,在他们出发之前,都会向你问问能不能到达他们想去的部落。

    题目描述

    简单起见,你就是要处理下面三件事,所有的事件都是按照时间顺序给出的。
    1.((Q) (p) (q) )从第 (p) 个部落出发的建筑工人想知道能否到达第 (q) 个部落了,你要回答的便是((Yes/No)),注意大小写
    2.((C) (p) (q) )第 (p) 个部落与第 (q) 个部落开战了,保证他们一定是相邻的部落,且目前处于停战(未开战)状态
    3.((U) (x) ) 第 (x) 次发生的战争结束了,它将永远的被载入史册,不复存在(保证这个消息不会告诉你多次)

    输入格式:

    第一行两个数 (n)(m)(n) 代表了一共有 (n) 个部落,(m) 代表了以上三种事件发生的总数
    接下来的 (n−1) 行,每行两个数 (p) , (q) ,代表了第 (p) 个部落与第 (q) 个部落之间有一条道路相连
    接下来的 (m) 行,每行表示一件事,详见题目描述

    输出格式:

    每行一个(“Yes”)或者(“No”),表示从第 (p) 个部落出发的建筑工人能否到达第 (q) 个部落

    输入样例#1:

    5 9
    1 2
    2 3
    3 4
    4 5
    Q 1 4
    C 2 1
    C 4 3
    Q 3 1
    Q 1 5
    U 1
    U 2
    C 4 3
    Q 3 4

    输出样例#1:

    Yes
    No
    No
    No

    输入样例#2:

    10 10
    1 2
    1 3
    3 4
    3 5
    1 6
    3 7
    1 8
    2 9
    5 10
    C 8 1
    Q 6 1
    C 2 1
    Q 2 10
    U 1
    C 9 2
    C 7 3
    U 3
    Q 6 7
    Q 1 10

    输出样例#2:

    Yes
    No
    No
    Yes

    输入样例#3:

    20 20
    1 2
    1 3
    2 4
    1 5
    1 6
    4 7
    1 8
    2 9
    5 10
    1 11
    2 12
    7 13
    1 14
    1 15
    11 16
    4 17
    3 18
    18 19
    8 20
    Q 13 5
    C 14 1
    C 16 11
    U 1
    U 2
    C 20 8
    Q 7 1
    C 7 4
    Q 17 17
    Q 1 6
    C 16 11
    C 2 1
    Q 16 2
    U 3
    U 5
    U 6
    C 2 1
    C 6 1
    C 13 7
    C 11 1

    输出样例#3:

    Yes
    Yes
    Yes
    Yes
    No

    说明

    对于(30%)的数据 (1<=n,m<=6000)
    对于另(30%)的数据,保证部落之间的地理关系是一条链,且 (i)(i+1) 之间有一条道路
    对于另(30%)的数据,(1<=n,m<=100000)
    对于(100%)的数据,(1<=n,m<=300000)


    树剖模板,线段树维护区间与运算之和,注意读入,其余无难度

    #define RG register
    #include<cstdio>
    #include<iostream>
    using namespace std;
    const int N=300005;
    inline int read()
    {
        RG int x=0,w=1;RG char ch=getchar();
        while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}
        while(ch>='0'&&ch<='9')x=x*10+ch-'0',ch=getchar();
        return x*w;
    }
    int n,m,cnt,ct,tot;
    int last[N],dep[N],dfn[N],top[N],size[N],fa[N],son[N],u[N],v[N];
    bool pass[N<<2];
    struct edge{int to,next;}e[N<<1];
    inline void insert(int u,int v)
    {
        e[++cnt]=(edge){v,last[u]};last[u]=cnt;
        e[++cnt]=(edge){u,last[v]};last[v]=cnt;
    }
    void dfs1(int now)
    {
        size[now]=1;
        for(int i=last[now];i;i=e[i].next)
        {
            int v=e[i].to;
            if(v==fa[now])continue;
            fa[v]=now;
            dep[v]=dep[now]+1;
            dfs1(v);
            size[now]+=size[v];
            if(size[v]>size[son[now]])son[now]=v;
        }
    }
    void dfs2(int now,int Top)
    {
        top[now]=Top;dfn[now]=++ct;
        if(son[now])dfs2(son[now],Top);
        for(int i=last[now];i;i=e[i].next)
        {
            int v=e[i].to;
            if(v==fa[now]||v==son[now])continue;
            dfs2(v,v);
        }
    }
    inline void Pushup(int root){pass[root]=pass[root<<1]&pass[root<<1|1];}
    void Build(int root,int l,int r)
    {
    	if(l==r){pass[root]=1;return;}
    	int mid=(l+r)>>1;
    	Build(root<<1,l,mid);
    	Build(root<<1|1,mid+1,r);
    	Pushup(root);
    }
    void Modify(int root,int l,int r,int s,int k)
    {
    	if(l==s&&r==s){pass[root]=k;return;}
    	int mid=(l+r)>>1;
    	if(s<=mid)Modify(root<<1,l,mid,s,k);
    	else Modify(root<<1|1,mid+1,r,s,k);
    	Pushup(root);
    }
    bool Query(int root,int l,int r,int ll,int rr)
    {
    	if(ll<=l&&r<=rr)return pass[root];
    	int mid=(l+r)>>1;
    	bool Ans=true;
    	if(mid>=ll)Ans&=Query(root<<1,l,mid,ll,rr);
    	if(mid<rr)Ans&=Query(root<<1|1,mid+1,r,ll,rr);
    	return Ans;
    }
    inline bool Query_Tree(int x,int y)
    {
    	bool Ans=true;
    	while(top[x]!=top[y])
    	{
    		if(dep[top[x]]<dep[top[y]])swap(x,y);
    		Ans&=Query(1,1,n,dfn[top[x]],dfn[x]);
    		x=fa[top[x]];
    	}
    	if(dep[x]>dep[y])swap(x,y);
    	if(x!=y)Ans&=Query(1,1,n,dfn[x]+1,dfn[y]);
    	return Ans;
    }
    int main()
    {
    	n=read();m=read();
    	for(int i=1;i<n;i++)
    	{
    		int p=read(),q=read();
    		insert(p,q);
    	}
    	dep[1]=1;
    	dfs1(1);
    	dfs2(1,1);
    	Build(1,1,n);
    	RG char xx=getchar();
    	while(m--)
    	{
    		while(xx!='Q'&&xx!='U'&&xx!='C')xx=getchar();
    		if(xx=='Q')
    		{
    			int p=read(),q=read();
    			if(Query_Tree(p,q))printf("Yes
    ");
    			else printf("No
    ");
    		}
    		if(xx=='U')
    		{
    			int x=read();
    			Modify(1,1,n,dfn[u[x]],1);
    		}
    		if(xx=='C')
    		{
    			tot++;
    			u[tot]=read();
    			v[tot]=read();
    			if(dfn[u[tot]]<dfn[v[tot]])swap(u[tot],v[tot]);
    			Modify(1,1,n,dfn[u[tot]],0);
    		}
    	}
    	return 0;
    }
    
  • 相关阅读:
    为什么和什么是 DevOps?
    使用jmeter 上传文件
    jmeter 获取执行脚本的路径
    随笔(九)
    随笔(八)
    随笔(七)
    随笔(六)
    随笔(五)
    随笔(四)
    随笔(三)
  • 原文地址:https://www.cnblogs.com/sdzwyq/p/8484167.html
Copyright © 2011-2022 走看看