评分稍微有一点过分。。不过这个题目确确实实很厉害,对思维训练也非常有帮助。

按照套路,我们把矩阵中的子段和化为前缀和相减的形式。题目就变成了给定一些前缀和之间的大小关系,让你构造一组可行的数据。这个东西显然是传递闭包,但是如果仅仅传递闭包的话我们只能询问两两之间的关系,处理起来比较麻烦。有没有什么办法获取一个完整的顺序序列出来呢?显然是做一个拓扑排序就好啦~

但是问题来了。如果没有相等一切好说,现在有相等该咋玩?

注意到题目保证一定有一组可行解,我们可以先不表示相等的关系。(如果把\(<=\)的关系和\(<\)的关系混起来会乱的~环什么的也不好处理)先把不等关系列出来,求出大小顺序的拓扑序列。由于一定有可行解,所以如果两个数相等的话,受到其他条件制约后,它们一定还是相邻的。这里用并查集维护一下相等关系,查一查前后相等不相等就可以了。

我们初始维护的是一个顺着边的方向递增的数量关系,\(DF
S\)求出的拓扑序列数量关系是递减的。所以处理的时候如果前后\(find\)一样就赋一样的值,否则就是前面等于后面\(+1/\)后面等于前面\(-1\)。求出\(Sum\)数组之后原数列就有了。

(特判:手推可知不连通的子图不会对答案造成影响。不用特判。)

#include <bits/stdc++.h>
using namespace std; const int N = 10 + 5; int T, n, sum[N], Set[N], vis[N]; char s[N * N]; vector <int> G[N], topo; int find (int x) {
return x == Set[x] ? x : Set[x] = find (Set[x]);
} void dfs (int u) {
vis[u] = true;
for (int i = 0; i < (int)G[u].size (); ++i) {
int v = G[u][i];
if (!vis[v]) dfs (v); //题目保证有解 = 无环
}
topo.push_back (u);
} int main () {
// freopen ("data.in", "r", stdin);
cin >> T;
while (T--) {
cin >> n >> s; int k = 0;
for (int i = 0; i < N; ++i) {
Set[i] = i, G[i].clear ();
}
for (int i = 1; i <= n; ++i) {
for (int j = i; j <= n; ++j) {
if (s[k] == '+') {//sum[j] > sum[i - 1]
G[i - 1].push_back (j);
}
if (s[k] == '-') {//sum[j] < sum[i - 1]
G[j].push_back (i - 1);
}
if (s[k] == '0') {
Set[find (j)] = find (i - 1);//数值相等 -> 同一个集合中
}
//x -> y : sum[x] < sum[y]
k = k + 1;
}
}
topo.clear ();
memset (vis, 0, sizeof (vis));
memset (sum, 0, sizeof (sum));
for (int i = 0; i <= n; ++i) {
if (!vis[i]) dfs (i); //简便的DFS拓扑排序23333
}
int zero_pos = -1;
for (int i = 0; i < (int) topo.size (); ++i) {
if (topo[i] == 0) {zero_pos = i; break;}
}
//找到0,前面是负数后面是正数
for (int i = zero_pos - 1; i >= 0; --i) {
if (find (topo[i]) == find (topo[i + 1])) {
sum[topo[i]] = sum[topo[i + 1]];//如果相等 -> 按照其他约束条件排出来也应该相邻
} else {
sum[topo[i]] = sum[topo[i + 1]] + 1;
}
}
for (int i = zero_pos + 1; i <= n; ++i) {
if (find (topo[i]) == find (topo[i - 1])) {
sum[topo[i]] = sum[topo[i - 1]];//如果相等 -> 按照其他约束条件排出来也应该相邻
} else {
sum[topo[i]] = sum[topo[i - 1]] - 1;
}
}
for (int i = 1; i <= n; ++i) {
cout << sum[i] - sum[i - 1];
if (i != n) cout << " ";
}
cout << endl;
}
}

LA4255/UVa1423 Guess 拓扑排序 并查集的更多相关文章

  1. ACM: hdu 1811 Rank of Tetris - 拓扑排序-并查集-离线

    hdu 1811 Rank of Tetris Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & % ...

  2. HDU 1811 拓扑排序 并查集

    有n个成绩,给出m个分数间的相对大小关系,问是否合法,矛盾,不完全,其中即矛盾即不完全输出矛盾的. 相对大小的关系可以看成是一个指向的条件,如此一来很容易想到拓扑模型进行拓扑排序,每次检查当前入度为0 ...

  3. 拓扑排序 - 并查集 - Rank of Tetris

    Description 自从Lele开发了Rating系统,他的Tetris事业更是如虎添翼,不久他遍把这个游戏推向了全球. 为了更好的符合那些爱好者的喜好,Lele又想了一个新点子:他将制作一个全球 ...

  4. LA 4255 (拓扑排序 并查集) Guess

    设这个序列的前缀和为Si(0 <= i <= n),S0 = 0 每一个符号对应两个前缀和的大小关系,然后根据这个关系拓扑排序一下. 还要注意一下前缀和相等的情况,所以用一个并查集来查询. ...

  5. Rank of Tetris(hdu1811拓扑排序+并查集)

    题意:关于Rating的信息.这些信息可能有三种情况,分别是"A > B","A = B","A < B",分别表示A的Rati ...

  6. Codeforces #541 (Div2) - D. Gourmet choice(拓扑排序+并查集)

    Problem   Codeforces #541 (Div2) - D. Gourmet choice Time Limit: 2000 mSec Problem Description Input ...

  7. hdu 1811 Rank of Tetris - 拓扑排序 - 并查集

    自从Lele开发了Rating系统,他的Tetris事业更是如虎添翼,不久他遍把这个游戏推向了全球. 为了更好的符合那些爱好者的喜好,Lele又想了一个新点子:他将制作一个全球Tetris高手排行榜, ...

  8. hdu 1811(缩点+拓扑排序+并查集)

    Rank of Tetris Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  9. hdu1811 Rank of Tetris 拓扑排序+并查集

    这道题是拓扑排序和并查集的综合运用. 由于排行榜是一种从高到低的排序.所以在拓扑排序的时候,如果有一次加入的入度为零的点数大于1,就有变得不确定了(UNCERTAIN). 由于只有一棵树,当树的数量大 ...

随机推荐

  1. Linux常用查找命令

    第一种:grep命令 示例: 第二种:find命令 示例: 第三种:locate命令 示例: 第四种:whereis命令 示例: 第五种:which命令 示例:

  2. java:数据结构(二)栈的应用(括号匹配)

    一.什么是括号匹配: 括号匹配就是利用计算机辨别表达式里面的括号是否书写成功 例如: {()((a)) }这就是一个正确 (()()   这就是一个错误的 二.括号匹配的算法: 众所周知,括号分为花括 ...

  3. git开发常用命令

    1.基本命令git branch 查看本地分支git branch -r 查看远程分支git checkout xxx 切换分支git pull origin master //从远程同步到本地,ma ...

  4. MongoDB个人简单总结

    当同一个变量被连续插入两次会出现id相同的异常,但是同样内容的两个变量同时插入不会有问题,可能是同一个变量同一个地址生成id相同,导致异常. 默认登陆不需要用户名密码认证,需要密码认证登陆需要在adm ...

  5. bash: lspci: command not found解决方法

    在CentOS虚拟机使得lspci查看硬件信息.使用时,提示bash: lspci: command not found,大多使用/sbin/lspci即可,我发现我的系统中/sbin下也没有.使用y ...

  6. 【shell基础】if分支语句

    1.if判断式if [ 条件判断一 ] && (||) [ 条件判断二 ]; thenelif [ 条件判断三 ] && (||) [ 条件判断四 ]; thenels ...

  7. [OIDC in Action] 3. 基于OIDC(OpenID Connect)的SSO(添加Github OAuth 2.0的支持)

    在上上一篇基于OIDC的SSO的登录页面的截图中有出现QQ登录的地方.这个其实是通过扩展OIDC的OpenID Provider来实现的,OpenID Provider简称OP,OP是OIDC的一个很 ...

  8. springboot项目从硬盘指定位置读取文件(获取静态资源)

    方法一:继承WebMvcConfigurerAdapter类 package com.imooc.demo.config; import org.springframework.context.ann ...

  9. 米卡 Mica Logo 存放处

  10. list遍历时删除的坑

    1.测试代码如下: public class StrTest { public static void main(String[] args) { ArrayList<String> li ...