hdu2094-产生冠军

题目链接

                                              产生冠军

Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 22104    Accepted Submission(s): 10040

 

Problem Description

有一群人,打乒乓球比赛,两两捉对撕杀,每两个人之间最多打一场比赛。
球赛的规则如下:
如果A打败了B,B又打败了C,而A与C之间没有进行过比赛,那么就认定,A一定能打败C。
如果A打败了B,B又打败了C,而且,C又打败了A,那么A、B、C三者都不可能成为冠军。
根据这个规则,无需循环较量,或许就能确定冠军。你的任务就是面对一群比赛选手,在经过了若干场撕杀之后,确定是否已经实际上产生了冠军。

 

Input

输入含有一些选手群,每群选手都以一个整数n(n<1000)开头,后跟n对选手的比赛结果,比赛结果以一对选手名字(中间隔一空格)表示,前者战胜后者。如果n为0,则表示输入结束。

 

Output

对于每个选手群,若你判断出产生了冠军,则在一行中输出“Yes”,否则在一行中输出“No”。

 

Sample Input

3

Alice Bob

Smith John

Alice Smith

5

a c

c d

d e

b e

a d

0

Sample Output

Yes

No

 

Author

qianneng

Source

迎接新学期——超级Easy版热身赛

思路:咋一看是拓扑排序,后来在比赛中终于想起来这个题目以前做过呀,就是没有输的人数减去输掉人数如果等于1就说明能成产生冠军。

显然图也是能写的,参考大神博客,就是有点麻烦。

wa代码:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<map>
#include<string>
#include<set>
#include<queue>
using namespace std;
priority_queue<int,vector<int> >q;
map<string,int>mp;
int main()
{
	int t,a,b;
	string str,str2;
	while(scanf("%d",&t),t){
		for(int i=0;i<t;i++){
			cin>>str>>str2;
			if(!mp[str]&&!mp[str2]){
				mp[str]=1000;
				mp[str2]=999;				
			}
			else if(!mp[str]&&mp[str2]){
				mp[str]=max(1000,mp[str2]+1);
			}
			else if(mp[str]&&!mp[str2]){
				mp[str2]=mp[str]-1;
			}
			else{
				int temp=mp[str2];
				mp[str2]=min(mp[str]-1,mp[str2]);
				mp[str]=max(temp+1,mp[str]);
			}
		}
		for(map<string,int>::iterator it=mp.begin();it!=mp.end();it++){
			int f=it->second;
			cout<<it->first<<" "<<it->second<<endl;
			q.push(f);
		}
		a=q.top();
		q.pop();
		b=q.top();
		q.pop();
		//cout<<"a : "<<a<<" b : "<<b<<endl;
		if(a!=b&&a==1000) printf("Yes\n");
		else printf("No\n");
		while(!q.empty()) q.pop();
		mp.clear();
	}
}

代码:

#include<iostream>
#include<cstdio>
#include<set>
using namespace std;
int main()
{
    string str,str2;
    int t;
    set<string>st;
    set<string>sp;
    while(scanf("%d",&t),t){//没有输的人数减去输掉人数如果等于1就说明能成产生冠军,否则
        for(int i=0;i<t;i++){
           cin>>str>>str2;
           st.insert(str);
           st.insert(str2);
           sp.insert(str2);     
        }
        if(st.size()-sp.size()==1) printf("Yes\n");
        else printf("No\n");
        st.clear();
        sp.clear();
    }
}

图代码:

#include <iostream> 
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <algorithm>
#include <cstring>
#include <string>
#include <set>
#include <map>
#include <queue>
#include <stack>
#include <vector>
#include <list>
#define rep(i,m,n) for(i=m;i<=n;i++)
#define rsp(it,s) for(set<int>::iterator it=s.begin();it!=s.end();it++)
const int inf_int = 2e9;
const long long inf_ll = 2e18;
#define inf_add 0x3f3f3f3f
#define mod 1000000007
#define vi vector<int>
#define pb push_back
//#define mp make_pair
#define fi first
#define se second
#define pi acos(-1.0)
#define pii pair<int,int>
#define Lson L, mid, rt<<1
#define Rson mid+1, R, rt<<1|1
const int maxn=5e2+10;
using namespace std;
typedef  long long ll;
typedef  unsigned long long  ull; 
inline int read(){int ra,fh;char rx;rx=getchar(),ra=0,fh=1;
while((rx<'0'||rx>'9')&&rx!='-')rx=getchar();if(rx=='-')
fh=-1,rx=getchar();while(rx>='0'&&rx<='9')ra*=10,ra+=rx-48,
rx=getchar();return ra*fh;}
//#pragma comment(linker, "/STACK:102400000,102400000")
ll gcd(ll p,ll q){return q==0?p:gcd(q,p%q);}
ll qpow(ll p,ll q){ll f=1;while(q){if(q&1)f=f*p;p=p*p;q>>=1;}return f;}
int n; 
set<string> s;
set<string>::iterator it;
int mpp[1005][1005];
vector<string> s1;
map<string,int> mp; 
int ct1;
int flag[1005];
int book[1005];
void dfs(int beg)
{
    ct1++;
    book[beg]=1;
    for(int i=1;i<=n;i++)
    {
        if(mpp[beg][i]==1&&book[i]==0)
        {
            dfs(i);
        }
    }
}
int main()
{
    string st;
    while(cin >>n)
    {
        if(n==0) break;
        memset(mpp,0,sizeof(mpp));
        memset(flag,0,sizeof(flag));
        s.clear();
        s1.clear();
        mp.clear();
        for(int i=0;i<n*2;i++)
        {
            cin >>st;
            s.insert(st);
            s1.push_back(st);
        }
        n++;
        int ct= 0;
        for(it=s.begin();it!=s.end();it++)
        {
            ct++;
            mp[*it] = ct;
        }
        for(int i=0;i<s1.size();i+=2)
        {
            mpp[mp[s1[i]]][mp[s1[i+1]]] =1;
        }
    	for(int i=1;i<s1.size();i+=2)
        {
        	flag[mp[s1[i]]]++;
    	}
    	n = s.size();
        int maxx=-1;
        for(int i=1;i<=n;i++)
        {
        	memset(book,0,sizeof(book));
            ct1=0;
            if(flag[i]) continue;
            dfs(i);
            if(maxx<ct1)
            {
                maxx=ct1;
            }
        }
        if(maxx==n)
            cout <<"Yes"<<endl;
        else
            cout <<"No"<<endl;
    }    
} 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值