zoukankan      html  css  js  c++  java
  • [codevs1027]姓名与ID

    [codevs1027]姓名与ID

    试题描述

    有N个人,各自有一个姓名和ID(别名)。每个人的姓名和ID都没有重复。这些人依次进入一间房间,然后可能会离开。过程中可以得到一些信息,告知在房间里的某个人的ID。你的任务是准确地确定每个人的ID。

    输入

    第一行是整数N,表示N个人,N<=20。

    接下来的一行是N个人的ID,用一个空格分隔。

    接下来的若干行是过程的记录:一个字母和一个字符串。字母是E、L或M中的一个。E表示进入房间,后面跟的字符串表示进来的人的姓名;L表示离开房间,后面跟的字符串表示离开的人的姓名;M表示回答询问,后面跟的字符串表示:当前用这个ID人在房间里面。

    最后一行Q表示结束。

    所有的姓名和ID都由不超过20个的小写字母组成。所有姓名都会在记录中出现。

    一开始时,房间时空的。

    输出

    共N行,每行形如:“姓名:ID”,如果ID不能确定,输出???。

    按照姓名的字典顺序输出。

    输入示例

    7
    bigman mangler sinbad fatman bigcheese frenchie capodicapo
    E mugsy
    E knuckles
    M bigman
    M mangler
    L mugsy
    E clyde
    E bonnie
    M bigman
    M fatman
    M frenchie
    L clyde
    M fatman
    E ugati
    M sinbad
    E moriarty
    E booth
    Q

    输出示例

    bonnie:fatman
    booth:???
    clyde:frenchie
    knuckles:bigman
    moriarty:???
    mugsy:mangler
    ugati:sinbad

    数据规模及约定

    见“输入”,还有操作数可以达到 10^5 左右。

    题解

    首先可以想到姓名和 ID 进行二分图匹配,现在难点在于判断唯一性。

    跑完二分图匹配后,我们逐个判断,对于一个姓名 i,我们强制不让它匹配它当前的匹配对象,在此基础上再跑一遍二分图,若还是完全匹配,则说明姓名 i 不能被唯一确定,标记成“???”。

    #include <iostream>
    #include <cstdio>
    #include <cstdlib>
    #include <cstring>
    #include <cctype>
    #include <algorithm>
    #include <string>
    #include <map>
    using namespace std;
    
    int read() {
    	int x = 0, f = 1; char c = getchar();
    	while(!isdigit(c)){ if(c == '-') f = -1; c = getchar(); }
    	while(isdigit(c)){ x = x * 10 + c - '0'; c = getchar(); }
    	return x * f;
    }
    
    #define maxn 45
    #define maxm 3210
    #define oo 2147483647
    struct Edge {
    	int from, to, flow;
    	Edge() {}
    	Edge(int _1, int _2, int _3): from(_1), to(_2), flow(_3) {}
    } ;
    struct Dinic {
    	int n, m, s, t, head[maxn], next[maxm];
    	Edge es[maxm];
    	int hd, tl, Q[maxn], vis[maxn];
    	int cur[maxn];
    	void cpy(Dinic& T) {
    		n = T.n; m = T.m; s = T.s; t = T.t;
    		memcpy(head, T.head, sizeof(head));
    		memcpy(next, T.next, sizeof(next));
    		for(int i = 0; i < m; i++) es[i] = T.es[i];
    		return ;
    	}
    	void init(int nn) {
    		n = nn; m = 0;
    		memset(head, -1, sizeof(head));
    		return ;
    	}
    	void AddEdge(int a, int b, int c) {
    		es[m] = Edge(a, b, c); next[m] = head[a]; head[a] = m++;
    		es[m] = Edge(b, a, 0); next[m] = head[b]; head[b] = m++;
    		return ;
    	}
    	bool BFS() {
    		memset(vis, 0, sizeof(vis)); vis[s] = 1;
    		hd = tl = 0; Q[++tl] = s;
    		while(hd < tl) {
    			int u = Q[++hd];
    			for(int i = head[u]; i != -1; i = next[i]) {
    				Edge& e = es[i];
    				if(!vis[e.to] && e.flow) {
    					vis[e.to] = vis[u] + 1;
    					Q[++tl] = e.to;
    				}
    			}
    		}
    		return vis[t] > 1;
    	}
    	int DFS(int u, int a) {
    		if(u == t || !a) return a;
    		int flow = 0, f;
    		for(int& i = cur[u]; i != -1; i = next[i]) {
    			Edge& e = es[i];
    			if(vis[e.to] == vis[u] + 1 && (f = DFS(e.to, min(a, e.flow)))) {
    				flow += f; a -= f;
    				e.flow -= f; es[i^1].flow += f;
    				if(!a) return flow;
    			}
    		}
    		return flow;
    	}
    	int MaxFlow(int ss, int tt) {
    		s = ss; t = tt;
    		int flow = 0;
    		while(BFS()) {
    			for(int i = 1; i <= n; i++) cur[i] = head[i];
    			flow += DFS(s, oo);
    		}
    		return flow;
    	}
    } sol, sol2, sol3;
    
    #define maxp 25
    string ID[maxp], name[maxp];
    map <string, int> M, M2;
    bool S[maxp];
    int en[maxn][maxn];
    struct Info {
    	string name, ID;
    	Info() {}
    	Info(string _, string __): name(_), ID(__) {}
    	bool operator < (const Info& t) const { return name < t.name; }
    } is[maxp];
    
    int main() {
    	int n = read();
    	sol.init((n << 1) + 2); int s = (n << 1) + 1, t = s + 1;
    	for(int i = 1; i <= n; i++) cin >> ID[i];
    	sort(ID + 1, ID + n + 1);
    	for(int i = 1; i <= n; i++) M[ID[i]] = i;
    	int cp = 0;
    	for(int i = 1; i <= n; i++)
    		for(int j = 1; j <= n; j++) en[i][j+n] = en[j+n][i] = 1;
    	while(1) {
    		char tc[2]; scanf("%s", tc);
    		if(tc[0] == 'Q') break;
    		string t; cin >> t;
    		if(tc[0] == 'E') {
    			if(!M2.count(t)) M2[t] = ++cp, name[cp] = t;
    			S[M2[t]] = 1;
    		}
    		if(tc[0] == 'L') S[M2[t]] = 0;
    		if(tc[0] == 'M') {
    			int p = M[t];
    			for(int i = 1; i <= n; i++) if(!S[i]) en[i][p+n] = en[p+n][i] = -1;
    		}
    	}
    	int ce = 0;
    	for(int i = 1; i <= cp; i++)
    		for(int j = n + 1; j <= (n << 1); j++) if(en[i][j] > 0)
    			sol.AddEdge(i, j, 1), en[i][j] = ce, ce += 2;
    	for(int i = 1; i <= cp; i++) sol.AddEdge(s, i, 1);
    	for(int i = 1; i <= n; i++) sol.AddEdge(i + n, t, 1);
    	sol3.cpy(sol); sol2.cpy(sol);
    	int ans = sol.MaxFlow(s, t);
    	for(int i = 1; i <= cp; i++) {
    		for(int j = n + 1; j <= (n << 1); j++) if(en[i][j] >= 0 && !sol.es[en[i][j]].flow) {
    			sol2.cpy(sol3);
    			sol2.es[en[i][j]].flow = 0;
    			int tmp = sol2.MaxFlow(s, t);
    //			printf("tmp: %d
    ", tmp);
    			if(tmp == ans) is[i] = Info(name[i], string("???"));
    			else is[i] = Info(name[i], ID[j-n]);
    			break;
    		}
    	}
    	
    	sort(is + 1, is + cp + 1);
    	for(int i = 1; i <= cp; i++) cout << is[i].name << ':' << is[i].ID << endl;
    	
    	return 0;
    }
    
  • 相关阅读:
    一个基于Angular+Ionic+Phonegap的混合APP实战
    Kafka
    kafka
    Kafka
    归并排序
    插入排序
    CC1310的晶振问题
    CC1310的RSSI值问题
    CC1310之使用SMARTRF STUDIO
    CC1310电源
  • 原文地址:https://www.cnblogs.com/xiao-ju-ruo-xjr/p/6036815.html
Copyright © 2011-2022 走看看