zoukankan      html  css  js  c++  java
  • CodeForces-1131D

    D. Gourmet choice

    题目链接:https://codeforces.com/problemset/problem/1131/D

    题目大意:NULL

    解题思路:如过两个菜之间是等号的话,则他们的序号应该相同,因此我们需要进行缩点操作,

    这里需要用到并查集,然后就是判断菜之间的顺序,因为他们之间有一个关系,我们可以考虑在两点之间建边,这时候就需要用到

    拓扑排序即可

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    const int maxn=1e4+10;
    int pre[maxn],in[maxn],out[maxn],vis[maxn],ans[maxn];
    vector<int>g[maxn];
    int find(int i)
    {
      if(i==pre[i]) return pre[i];
      else return find(pre[i]);
    }
    void unions(int x,int y)
    {
        int xx=find(x);
        int yy=find(y);
        if(xx!=yy) pre[xx]=yy;
    }
    int main()
    {
        int n,m;
        cin>>n>>m;
        for(int i=1;i<=n+m;i++)
        {
            pre[i]=i;
        }
        char s[1100][1100];
        for(int i=1;i<=n;i++)
        {
            scanf("%s",s[i]+1);
            for(int j=1;j<=m;j++)
            {
                if(s[i][j]=='=') unions(i,j+n);
                /*else if(s[i][j]=='>') g[find(j+n)].push_back(find(i)),in[find(i)]++;
                else g[find(i)].push_back(find(j+n)),in[find(j+n)]++;*/
            }
        }
         for(int i=1; i<=n; i++){
            for(int j=1; j<=m; j++)
            if(s[i][j] == '>'){
               g[find(j+n)].push_back(find(i)); in[find(i)]++;
            }
            else if(s[i][j] == '<') g[find(i)].push_back(find(j+n)),in[find(j+n)]++;
        }
        stack<int>q;
        for(int i=1;i<=n+m;i++)
        {
            if(in[find(i)]==0)
            {
            //    cout<<find(i)<<endl;
                q.push(find(i));
                vis[find(i)]=1;
                ans[find(i)]=1;
            }
        }
        while(!q.empty())
        {
            int u=q.top();
            q.pop();
            for(int i=0;i<g[u].size();i++)
            {
                int v=g[u][i];
                in[find(v)]--;
                if(in[find(v)]==0&&!vis[find(v)])
                {
                     vis[find(v)]=1;
                     ans[find(v)]=ans[find(u)]+1;
                     q.push(find(v));
                }
            }
        }
        for(int i=1;i<=n+m;i++)
        {
            if(!vis[find(i)])
            {
                cout<<"NO
    ";
                return 0;
            }
        }
        cout<<"YES"<<endl;
        for(int i=1;i<=n;i++)
        {
            if(i!=1) cout<<" ";
            cout<<ans[find(i)]; 
        }
        cout<<endl;
        for(int i=1;i<=m;i++)
        {
            if(i!=1) cout<<" ";
            cout<<ans[find(i+n)]; 
        }
        return 0;
    }
  • 相关阅读:
    常用的文件查看命令
    Linux常用快捷按键
    寒冬储粮
    创建型模式:抽象工厂
    创建型模式:工厂方法
    创建型模式:单例模式
    开闭原则
    迪米特法则
    接口隔离原则
    依赖倒置原则
  • 原文地址:https://www.cnblogs.com/tombraider-shadow/p/11622494.html
Copyright © 2011-2022 走看看