zoukankan      html  css  js  c++  java
  • [CF466E] Information Graph

    Description

    公司中有 (n) 名员工,开始时员工之间没有任何联系,接下来会依次发生 (m) 条事件,类型有三种。

    1. (y) 成了 (x) 的上司
    2. (x) 得到了一份文件,(x) 把文件传给了他的上司,他的上司又会传给上司的上司,直到某人没有上司,此时文件被销毁
    3. 询问 (x) 是否看过第 (i) 份文件

    Solution

    考虑离线处理,时序扫描,第一轮处理 1,2 操作,第二轮处理询问

    对于操作 1,在并查集中合并 (x,y)(顺序!),在森林中加边

    对于操作 2,我们将这份文件记录下来,其信息为 (x) 和它在并查集中的根

    时序扫描过程中,设第 (i) 份文件涉及的两个点是 (p_i,q_i),我们只需要检查 (x) 是否在 (p,q) 构成的链上即可

    #include <bits/stdc++.h>
    using namespace std;
    
    #define int long long
    const int N = 1000005;
    const int M = 20;
    
    int n,m,t1,t2,t3;
    
    struct Forest
    {
        int fa[N][M],dep[N],vis[N],deg[N];
        vector <int> g[N];
    
        void make(int p,int q)
        {
            g[q].push_back(p);
            deg[p]++;
        }
    
        void dfs(int p)
        {
            vis[p]=1;
            for(int q:g[p])
            {
                dep[q]=dep[p]+1;
                fa[q][0]=p;
                dfs(q);
            }
        }
    
        void presolve()
        {
            for(int i=1;i<=n;i++)
            {
                if(deg[i]==0)
                {
                    dep[i]=1;
                    dfs(i);
                }
            }
            for(int j=1;j<M;j++)
            {
                for(int i=1;i<=n;i++)
                {
                    fa[i][j]=fa[fa[i][j-1]][j-1];
                }
            }
        }
    
        bool check_relation(int p,int q)
        {
            for(int i=M-1;i>=0;--i)
            {
                if(dep[fa[p][i]]>=dep[q]) p=fa[p][i];
            }
            return p==q;
        }
    
        bool query(int p,int q,int v)
        {
            return check_relation(p,v) && check_relation(v,q);
        }
    } forest;
    
    struct Dsu
    {
        int fa[N];
    
        void init()
        {
            for(int i=1;i<=n;i++) fa[i]=i;
        }
    
        int find(int p)
        {
            return fa[p]==p ? p : fa[p]=find(fa[p]);
        }
    
        void merge(int p,int q)
        {
            p=find(p);
            q=find(q);
            if(p!=q) fa[p]=q;
        }
    } dsu;
    
    struct Document
    {
        int p,q;
    } doc[N];
    
    struct Request
    {
        int x,i;
    } req[N];
    
    int nreq;
    
    int ndoc;
    
    signed main()
    {
        ios::sync_with_stdio(false);
    
        cin>>n>>m;
    
        dsu.init();
    
        for(int i=1;i<=m;i++)
        {
            cin>>t1;
    
            if(t1==1)
            {
                cin>>t2>>t3;
                forest.make(t2,t3);
                dsu.merge(t2,t3);
            }
            else if(t1==2)
            {
                cin>>t2;
                doc[++ndoc]={t2,dsu.find(t2)};
            }
            else if(t1==3)
            {
                cin>>t2>>t3;
                req[++nreq]={t2,t3};
                if(dsu.find(t2) != dsu.find(doc[t3].p)) req[nreq].x=-1;
            }
        }
    
        forest.presolve();
    
        for(int u=1;u<=nreq;u++)
        {
            int x=req[u].x, i=req[u].i;
            if(x>0)
            {
                if(forest.query(doc[i].p,doc[i].q,x))
                {
                    cout<<"YES"<<endl;
                }
                else
                {
                    cout<<"NO"<<endl;
                }
            }
            else
            {
                cout<<"NO"<<endl;
            }
        }
    }
    
    
  • 相关阅读:
    C#调用自定义表类型参数
    不同版本SQL SERVER备份还原时造成索引被禁用
    SQL SERVER同步环境新增发布对象时不能生成(sp_MS+表名)同步存储过程
    C# 读取在存储过程多结果集
    C#读取XML文件
    批量还原V2
    tmux 常用快捷键
    无法生成SSPI上下文
    sql server 性能计数器
    sql server 2008 r2 xevent
  • 原文地址:https://www.cnblogs.com/mollnn/p/13589777.html
Copyright © 2011-2022 走看看