Bzoj : 1823: [JSOI2010]滿漢全席
1823: [JSOI2010]滿漢全席
Description
滿漢全席是中國最豐盛的宴客菜肴,有許多種不同的材料透過滿族或是漢族的料理方式,呈現在數量繁多的菜色之中。由於菜色眾多而繁雜,只有極少數博學多聞技藝高超的廚師能夠做出滿漢全席,而能夠烹飪出經過專家認證的滿漢全席,也是中國廚師最大的榮譽之一。 世界滿漢全席協會是由能夠料理滿漢全席的專家廚師們所組成,而他們之間還細分為許多不同等級的廚師。為了招收新進的廚師進入世界滿漢全席協會,將於近日舉辦滿漢全席大賽,協會派遣許多會員當作評審員,為的就是要在參賽的廚師之中,找到滿漢料理界的明日之星。 大會的規則如下:每位參賽的選手可以得到n 種材料,選手可以自由選擇用滿式或是漢式料理將材料當成菜肴。大會的評審制度是:共有m 位評審員分別把關。每一位評審員對於滿漢全席有各自獨特的見解,但基本見解是,要有兩樣菜色作為滿漢全席的標誌。如某評審認為,如果沒有漢式東坡肉跟滿式的涮羊肉鍋,就不能算是滿漢全席。但避免過於有主見的審核,大會規定一個評審員除非是在認為必備的兩樣菜色都沒有做出來的狀況下,才能淘汰一位選手,否則不能淘汰一位參賽者。換句話說,只要參賽者能在這兩種材料的做法中,其中一個符合評審的喜好即可通過該評審的審查。如材料有豬肉,羊肉和牛肉時,有四位評審員的喜好如下表: 評審一 評審二 評審三 評審四 滿式牛肉 滿式豬肉 漢式牛肉 漢式牛肉 漢式豬肉 滿式羊肉 漢式豬肉 滿式羊肉 如參賽者甲做出滿式豬肉,滿式羊肉和滿式牛肉料理,他將無法滿足評審三的要求,無法通過評審。而參賽者乙做出漢式豬肉,滿式羊肉和滿式牛肉料理,就可以滿足所有評審的要求。 但大會後來發現,在這樣的制度下如果材料選擇跟派出的評審員沒有特別安排好的話,所有的參賽者最多只能通過部分評審員的審查而不是全部,所以可能會發生沒有人通過考核的情形。如有四個評審員喜好如下表時,則不論參賽者采取什麽樣的做法,都不可能通過所有評審的考核: 評審一 評審二 評審三 評審四 滿式羊肉 滿式豬肉 漢式羊肉 漢式羊肉 漢式豬肉 滿式羊肉 漢式豬肉 滿式豬肉 所以大會希望有人能寫一個程序來判斷,所選出的m 位評審,會不會發生 沒有人能通過考核的窘境,以便協會組織合適的評審團。
Input
第一行包含一個數字 K,代表測試文件包含了K 組資料。每一組測試資料的第一行包含兩個數字n 跟m(n≤100,m≤1000),代表有n 種材料,m 位評審員。為方便起見,材料舍棄中文名稱而給予編號,編號分別從1 到n。接下來的m 行,每行都代表對應的評審員所擁有的兩個喜好,每個喜好由一個英文字母跟一個數字代表,如m1 代表這個評審喜歡第1 個材料透過滿式料理做出來的菜,而h2 代表這個評審員喜歡第2 個材料透過漢式料理做出來的菜。每個測試文件不會有超過50 組測試資料
Output
每筆測試資料輸出一行,如果不會發生沒有人能通過考核的窘境,輸出GOOD;否則輸出BAD(大寫字母)。
Sample Input
23 4
m3 h1
m1 m2
h1 h3
h3 m2
2 4
h1 m2
m2 m1
h1 h2
m1 h2
Sample Output
GOOD
BAD
思路 :
2 - SAT, 比較裸, 2sat連邊之後, 跑個Tarjan, 如果所有食材的兩種做法都不在一個強連通分量裏,則可以做出來, 否則BAD
#include <cstdio> #include <cstring> #include <algorithm> #include <iostream> #include<stack> #include <cctype> #define ms(a) (memset(a, 0, sizeof(a))) #define min(a, b) (a<b?a:b) using namespace std; const int N = 1110, M = 21100; int head[N], to[M], nxt[M], cnt; int dfn[N], low[N], place[N], tot; int ins[N], idx; stack<int>s; inline char nc() { static char buf[100000], *p1, *p2; return p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, 100000 ,stdin), p1==p2)?EOF:*p1++; } inline char gc() { char c = nc(); while(isspace(c)) c = nc(); return c; } inline int read() { int x = 0; char c = nc(); while(!isdigit(c))c=nc(); while(isdigit(c)) {x=(x<<3)+(x<<1)+(c^48), c=nc();} return x; } void init() { ms(dfn); ms(head); ms(low); cnt = tot = 0; ms(place); while(!s.empty())s.pop(); } void add(int x, int y) { to[++cnt] = y; nxt[cnt] = head[x]; head[x] = cnt; } void Tarjan(int p) { dfn[p] = low[p] = ++idx; s.push(p); ins[p] = 1; //puts("FUCK"); for(int i=head[p];i;i=nxt[i]) { if(!dfn[to[i]]) { Tarjan(to[i]); low[p] = min(low[p], low[to[i]]); } else if(ins[to[i]]) { low[p] = min(low[p], dfn[to[i]]); } } if(low[p]==dfn[p]) { tot++; int t = 0; while(t!=p) { t = s.top();s.pop(); ins[t] = 0; place[t] = tot; } } } int main() { int t = read(); while(t--) { init(); int n = read(), m = read(); for(int i=1;i<=m;i++) { char k1, k2; int x1, x2, y1=0, y2=0; k1 = gc(), x1 = read(), k2 = gc(), x2 = read(); if(k1 == ‘m‘) y1 = 1; if(k2 == ‘m‘) y2 = 1; add((1-y2)*n+x2, y1*n+x1); add((1-y1)*n+x1, y2*n+x2); } for(int i=1;i<=n*2;i++) if(!dfn[i]) Tarjan(i); bool flag = 0; for(int i=1;i<=n;i++) if(place[i]==place[n+i]) flag = 1; puts(flag?"BAD":"GOOD"); } }
Bzoj : 1823: [JSOI2010]滿漢全席