• 【codeforces 505D】Mr. Kitayuta's Technology


    【题目链接】:http://codeforces.com/problemset/problem/505/D

    【题意】

    让你构造一张有向图;
    n个点;
    以及所要求的m对联通关系(xi,yi)
    即要求这张有向图中的点xi能够联通到点yi;
    问你最少需要添加多少条边才够;

    【题解】

    先将输入的m条边;
    当成无向边,构成一张无向图;
    然后对于构成这张图的各个联通块;
    设len为这个联通块的节点个数;
    如果这个联通块它对应的有向图内有环;
    则这个联通块需要len条有向边;
    (即这len个节点首尾相连构成一个环,只需要len条边)
    这样不管你内部要怎么样的连通性都行,因为任意两个点都是联通的;
    如果对应的有向图没环;
    则这个联通块只需要len-1条有向边;
    (总能用len-1条边构造出来符合要求的图的..因为没有环)
    把各个联通块的答案都累加起来就好;
    有向图找环用拓扑排序就好;
    (防止爆栈什么的 。)

    【Number Of WA

    0

    【完整代码】

    #include <bits/stdc++.h>
    using namespace std;
    #define lson l,m,rt<<1
    #define rson m+1,r,rt<<1|1
    #define LL long long
    #define rep1(i,a,b) for (int i = a;i <= b;i++)
    #define rep2(i,a,b) for (int i = a;i >= b;i--)
    #define mp make_pair
    #define pb push_back
    #define fi first
    #define se second
    #define ms(x,y) memset(x,y,sizeof x)
    #define Open() freopen("F:\rush.txt","r",stdin)
    #define Close() ios::sync_with_stdio(0),cin.tie(0)
    
    typedef pair<int,int> pii;
    typedef pair<LL,LL> pll;
    
    const int dx[9] = {0,1,-1,0,0,-1,-1,1,1};
    const int dy[9] = {0,0,0,-1,1,-1,1,-1,1};
    const double pi = acos(-1.0);
    const int N = 1e5+100;
    
    int n,m,rudu[N],ans;
    vector <int> G[N],G1[N],v;
    queue <int> dl;
    
    bool vis[N];
    
    void dfs(int x){
        if (vis[x]) return;
        vis[x] = true;
        v.pb(x);
        int len = G1[x].size();
        rep1(i,0,len-1)
            dfs(G1[x][i]);
    }
    
    int main(){
        //Open();
        Close();//scanf,puts,printf not use
        //init??????
        cin >> n >> m;
        rep1(i,1,m){
            int x,y;
            cin >> x >> y;
            G[x].pb(y);
            rudu[y]++;
            G1[x].pb(y);
            G1[y].pb(x);
        }
    
        rep1(i,1,n)
            if (!vis[i]){
                v.clear();
                dfs(i);
                while (!dl.empty()) dl.pop();
                rep1(j,0,(int) v.size()-1)
                    if (rudu[v[j]]==0){
                        dl.push(v[j]);
                    }
                int num = 0;
                while (!dl.empty()){
                    int x = dl.front();
                    num++;
                    dl.pop();
                    rudu[x] = -1;
                    rep1(j,0,(int) G[x].size()-1){
                        rudu[G[x][j]]--;
                        if (rudu[G[x][j]]==0){
                            dl.push(G[x][j]);
                        }
                    }
                }
                ans+=(int) v.size() - (num==(int) v.size() ? 1:0);
            }
        cout << ans << endl;
        return 0;
    }
    
  • 相关阅读:
    解析HTTP协议六种请求方法
    金蝶
    普元
    中间件
    [CTSC2008] 网络管理
    【Uva 10498】满意值
    【SPOJ839】最优标号
    bzoj2879 [Noi2012]美食节
    bzoj3144 [Hnoi2013]切糕
    bzoj3112 [Zjoi2013]防守战线
  • 原文地址:https://www.cnblogs.com/AWCXV/p/7626293.html
Copyright © 2020-2023  润新知