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

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

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

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

我们初始维护的是一个顺着边的方向递增的数量关系,\(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. flex-骰子布局

    弹性容器单行:主轴居中,交叉轴居中. display: flex; flex-direction: row; align-items: center; justify-content: center; ...

  2. arcgis api 3.x for js 入门开发系列二十一气泡窗口信息动态配置模板

    前言 关于本篇功能实现用到的 api 涉及类看不懂的,请参照 esri 官网的 arcgis api 3.x for js:esri 官网 api,里面详细的介绍 arcgis api 3.x 各个类 ...

  3. iOS---------获取当前年份

    NSDate *  senddate=[NSDate date]; NSDateFormatter  *dateformatter=[[NSDateFormatter alloc] init]; [d ...

  4. Chart.js 與 ASP.NET MVC 整合應用

    Chart.js 是一套開放原始碼的「圖表」繪製函式庫,和其他第三方的圖表工具相比,Chart.js 的特色如下: 支援 HTML 5.響應式網頁 (RWD, Responsive Web Desig ...

  5. Android视频录制从不入门到入门系列教程(三)————视频方向

    运行Android视频录制从不入门到入门系列教程(二)————显示视频图像中的Demo后,我们应该能发现视频的方向是错误的. 由于Android中,Camera给我们的视频图片的原始方向是下图这个样子 ...

  6. C# ToString()日期格式

    C# ToString()日期格式 ToString:2016/9/27 0:00:00ToString("yyyy/MM/dd"):2016/09/27ToString(&quo ...

  7. SQL语句将一个表的数据写入到另一个表中

    在日常的数据库运维过程中,有时候需要将Select查询出来的数据集写入到另一个数据表中,其中一种方式是通过存储过程循环写入数据,另一种简便的方式是直接使用Insert Into语句后面跟上Select ...

  8. c#面试题汇总(1)

    c#面试题汇总 From: https://www.cnblogs.com/suzhiyong1988/p/5069385.html   下面的参考解答只是帮助大家理解,不用背,面试题.笔试题千变万化 ...

  9. zabbix配置邮件告警

    1.安装邮件服务 yum -y install mailx @qq.com #发邮件测试 2.添加报警媒介 输入接收告警邮件的邮箱 3.配置SMTP服务端 使用本机作为SMTP服务器 4.创建触发器 ...

  10. 一个简易的kmp教学并给出java实现

    简单介绍一下问题 给定source字符串,找出target字符串出现的首位 例如 source   为“abddabddabc” target 为 “abddabc” 从第一位开始比较 |a b d ...