[JSOI2010]满汉全席

[JSOI2010]满汉全席

Time Limit:10000MS  Memory Limit:65536K
Total Submit:34 Accepted:21
Case Time Limit:1000MS

Description

满汉全席是中国最丰盛的宴客菜肴,有许多种不同的材料透过满族或是汉族的料理方式,呈现在數量繁多的菜色之中。由于菜色众多而繁杂,只有极少數博学多闻技 艺高超的厨师能够做出满汉全席,而能够烹饪出经过专家认证的满汉全席,也是中国厨师最大的荣誉之一。
世界满汉全席协会是由能够料理满汉全席的专家厨师们所组成,而他们之间还细分为许多不同等级的厨师。为了招收新进的厨师进入世界满汉全席协会,将 于近日举办满汉全席大赛,协会派遣许多会员当作评审员,为的就是要在參赛的厨师之中,找到满汉料理界的明日之星。
大会的规则如下:每位參赛的选手可以得到n 种材料,选手可以自由选择用满式或是汉式料理将材料当成菜肴。大会的评审制度是:共有m 位评审员分别把关。每一位评审员对于满汉全席有各自独特的見解,但基本见解是,要有兩样菜色作为满汉全席的标志。如某评审认为,如果没有汉式东坡肉跟满式 的涮羊肉锅,就不能算是满汉全席。但避免过于有主見的审核,大会规定一个评审员除非是在认为必备的两样菜色都没有做出來的狀况下,才能淘汰一位选手,否则 不能淘汰一位參赛者。换句话說,只要參赛者能在这兩种材料的做法中,其中一个符合评审的喜好即可通过该评审的审查。如材料有猪肉,羊肉和牛肉时,有四位评 审员的喜好如下表:
评审一 评审二 评审三 评审四
满式牛肉 满式猪肉 汉式牛肉 汉式牛肉
汉式猪肉 满式羊肉 汉式猪肉 满式羊肉
如參赛者甲做出满式猪肉,满式羊肉和满式牛肉料理,他将无法满足评审三的要求,无法通过评审。而參赛者乙做出汉式猪肉,满式羊肉和满式牛肉料理, 就可以满足所有评审的要求。
但大会后來发现,在这样的制度下如果材料选择跟派出的评审员没有特别安排好的话,所有的參赛者最多只能通过部分评审员的审查而不是全部,所以可能 会发生没有人通过考核的情形。如有四个评审员喜好如下表时,则不論參赛者采取什么样的做法,都不可能通过所有评审的考核:
评审一 评审二 评审三 评审四
满式羊肉 满式猪肉 汉式羊肉 汉式羊肉
汉式猪肉 满式羊肉 汉式猪肉 满式猪肉
所以大会希望有人能写一个程序來判断,所选出的m 位评审,会不会发生 没有人能通过考核的窘境,以便协会组织合适的评审团。

Input

第一行包含一个数字 K,代表测试文件包含了K 组资料。每一组测试资料的第一行包含兩个数字n 跟m(n≤100,m≤1000),代表有n 种材料,m 位评审员。为方便起見,材料舍弃中文名称而给予编号,编号分别从1 到n。接下來的m 行,每行都代表对应的评审员所拥有的兩个喜好,每个喜好由一个英文字母跟一个数字代表,如m1 代表这个评审喜欢第1 个材料透过满式料理做出來的菜,而h2 代表这个评审员喜欢第2 个材料透过汉式料理做出來的菜。每个测试文件不会有超过50 组测试资料,而每笔测试资料中材料的种类数跟评审员的个数均不超过15。

Output

每笔测试资料输出一行,如果不会发生没有人能通过考核的窘境,输出GOOD;否则输出BAD(大写字母)。

Sample Input

2
3 4
m3 h1
m1 m2
h1 h3
h3 m2
2 4
h1 m2
m2 m1
h1 h2
m1 h2

Sample Output

GOOD
BAD

Source

JSOI2010第二轮Contest2
水题啊。。一眼就看出是2-sat,而且还不需要方案囧。。直接构图然后Tarjan就可以了。。
具体说就是对每个菜建两个节点,一个节点连向另一节点表示选了这个就一定要选那个,比如一个人要求一个汉一个满,如果第一个选了满那么第二个只能是汉了(*^__^*) 。。然后看看有没有两个矛盾的量在一个强联通分量里的就可以了囧。。
Code:

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<string>
#include<vector>
#include<cstring>
#include<stack>
#define rep(i,n) for(int i=0;i<n;i++)
#define pb push_back
using namespace std;
const int inf=~0U>>1,maxn=100*2;
struct Edge
{
int t;Edge*next;
Edge(int _t,Edge* _next):t(_t),next(_next){}
}*E[maxn]={0};
void AddEdge(int s,int t){E[s]=new Edge(t,E[s]);}
int n,m;
inline int node(int i,bool j){return i*2+j;}
int low[maxn],ord[maxn],id[maxn],cnt,snt;bool instack[maxn];
stack<int> S;
int dfs(int x)
{
low[x]=ord[x]=++cnt;
S.push(x);instack[x]=true;
for(Edge*e=E[x];e;e=e->next)
if(!ord[e->t]) low[x]=min(low[x],dfs(e->t));
else if(instack[e->t]) low[x]=min(low[x],ord[e->t]);
if(low[x]==ord[x])
{
int u;
do{u=S.top();S.pop();id[u]=snt;instack[u]=false;}while(u!=x);
snt++;
}
return low[x];
}
void Tarjan()
{
memset(ord,0,sizeof(ord));
memset(instack,0,sizeof(instack));
cnt=snt=0;
rep(i,2*n)if(!ord[i])dfs(i);
}
bool Judge()
{
rep(i,n) if(id[2*i]==id[2*i+1])return false;
return true;
}
void solve()
{
scanf("%d %dn",&n,&m);bool c[2];int x[2];char t;
memset(E,0,sizeof(E));
while(m–)
{
rep(i,2)scanf("%c%dn",&t,x+i),–x[i],c[i]=t==’h';
rep(i,2) AddEdge(node(x[i],!c[i]),node(x[1-i],c[1-i]));
}
Tarjan();
if(Judge())cout<<"GOOD"<<endl;
else cout<<"BAD"<<endl;
}
int main()
{
//freopen("in","r",stdin);
int t;cin>>t;while(t–)solve();
}

3 thoughts on “[JSOI2010]满汉全席

Leave a Reply

Your email address will not be published. Required fields are marked *

You may use these HTML tags and attributes: <a href="" title=""> <abbr title=""> <acronym title=""> <b> <blockquote cite=""> <cite> <code> <del datetime=""> <em> <i> <q cite=""> <s> <strike> <strong>