思路:比较显然的做法是二分图匹配然而复杂度太高了,可以这样考虑,如果王子喜欢那个公主的话,那么连一条u->v+n的边,然后由最后的方案里如果王子娶了那个公主的话那么连一条v+n->u的边,然后求一次tarjan缩点就可以啦,为什么这样是可行的呢?
参考别人的思路:点击打开链接
#include <cstdio>
#include <queue>
#include <cstring>
#include <iostream>
#include <cstdlib>
#include <algorithm>
#include <vector>
#include <map>
#include <string>
#include <set>
#include <ctime>
#include <cmath>
#include <cctype>
#include <stack>
using namespace std;
#define maxn 4000+100
#define LL long long
int cas=1,T;
vector<int>G[maxn];
int pre[maxn];
int lowlink[maxn];
int sccno[maxn];
int dfs_clock,scc_cnt;
int n,m;
const int INF = 1<<29;
stack<int>S;
void dfs(int u)
{
pre[u]=lowlink[u]=++dfs_clock;
S.push(u);
for (int i = 0;i<G[u].size();i++)
{
int v = G[u][i];
if (!pre[v])
{
dfs(v);
lowlink[u] = min(lowlink[u],lowlink[v]);
}
else if (!sccno[v])
{
lowlink[u] = min (lowlink[u],pre[v]);
}
}
if (lowlink[u] == pre[u])
{
scc_cnt++;
for (;;)
{
int x = S.top();S.pop();
sccno[x] = scc_cnt;
if (x==u)
break;
}
}
}
void find_scc(int n)
{
dfs_clock=scc_cnt=0;
memset(sccno,0,sizeof(sccno));
memset(pre,0,sizeof(pre));
for (int i = 0;i<n;i++)
if (!pre[i])
dfs(i);
}
vector<int>ans;
int main()
{
while (scanf("%d",&n)!=EOF)
{
for (int i = 0;i<=2*n;i++)
G[i].clear();
for(int i = 1;i<=n;i++)
{
int num = 0;
scanf("%d",&num);
for(int j = 1;j<=num;j++)
{
int v;
scanf("%d",&v);
G[i].push_back(v+n);
}
}
for(int i = 1;i<=n;i++)
{
int x;
scanf("%d",&x);
G[x+n].push_back(i);
}
find_scc(n);
for(int u = 1;u<=n;u++)
{
for(int i = 0;i<G[u].size();i++)
{
if(sccno[u]==sccno[G[u][i]])
ans.push_back(G[u][i]);
}
sort(ans.begin(),ans.end());
if(ans.size()!=0)
{
printf("%d ",ans.size());
for(int i = 0;i<ans.size();i++)
printf("%d%c",ans[i]>n?ans[i]-n:ans[i],i==ans.size()-1?'\n':' ');
}
ans.clear();
}
}
return 0;
}