题目连接
题意
给你两个长度分别为 n,m的正整数序列 a,b。以矩阵形式给出所有 a i {a_i} ai 和 b j b_j bj(1≤i≤n,1≤j≤m) 的大小关系(>,<,=)。构造符合条件的 a 和 b序列,且 a,b 中最大元素尽量小。无解,输出No。
思路
将每个元素设为一个节点,两者元素如果相同则用并查集合并,最后对合并后的点建图。
如a>b,b到a存在一条单向边
a<b,a到b存在一条单向边
记录所有边出度,进行拓扑排序,按拓扑排序从1开始标号。对于一个集合的点标号相同。
有环无解,无环输出标号即可。
代码
#include <bits/stdc++.h>
using namespace std;
int f[2005], de[2005], ans[2005], book[2005];
char s[1005][1005];
vector<int> e[2005];
queue<int> q;
int getf(int a)
{
return a == f[a] ? a : f[a] = getf(f[a]);
}
int main()
{
int n, m;
scanf("%d%d",&n,&m);
for(int i = 1; i <= n+m; ++i) f[i] = i;
for(int i = 1; i <= n; ++i)
{
scanf("%s",s[i]+1);
for(int j = 1; j <= m; ++j)
if(s[i][j] == '=') f[getf(n+j)] = getf(i);
}
for(int i = 1; i <= n; ++i)
{
for(int j = 1; j <= m; ++j)
{
if(s[i][j] == '>') ++de[getf(i)], e[getf(n+j)].push_back(getf(i));
if(s[i][j] == '<') ++de[getf(n+j)], e[getf(i)].push_back(getf(n+j));
}
}
for(int i = 1; i <= n+m; ++i)
if(!de[getf(i)] && !book[getf(i)]) q.push(getf(i)), ans[getf(i)] = 1, book[getf(i)] = 1;
while(!q.empty())
{
int u = getf(q.front());
q.pop();
for(auto v : e[u])
{
if(--de[getf(v)] == 0 && !book[getf(v)])
{
book[getf(v)] = 1;
ans[getf(v)] = ans[u]+1;
q.push(getf(v));
}
}
}
for(int i = 1; i <= n+m; ++i) if(de[i]) return !printf("No\n");
printf("Yes\n");
for(int i = 1; i <= n+m; ++i) printf("%d%c",ans[getf(i)], i == n ? '\n' : ' ');
return 0;
}