BZOJ.4727.[POI2017]Turysta(哈密顿路径/回路 竞赛图)
\(Description\)
给出一个n个点的有向图,任意两个点之间有且仅一条有向边。对于每个点v,求出从v出发的一条经过点数最多,且没有重复经过同一个点一次以上的简单路径。
n<=2000
\(Solution\)
竞赛图缩点后得到的拓扑图一定是一条链,因为竞赛图任意两点前后关系确定,所以只有一种拓扑序列
从前边强连通分量中的任意一点出来 都可以到达后边强连通分量的任意一点
因为竞赛图的每个强连通分量一定存在一条哈密顿回路
所以只需要求出每一个强连通分量的哈密顿回路,然后在链上走,把每个强连通分量的回路存起来中,最后按拓扑序从后往前输出即可
//4884kb 9220ms
#include <cstdio>
#include <cctype>
#include <vector>
#include <algorithm>
#define gc() getchar()
const int N=2005;
int n,dfn[N],low[N],id,cnt,bel[N],sk[N],top,nxt[N],dgr[N],pos[N];
bool mp[N][N],ins[N];
std::vector<int> scc[N];
inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
void Tarjan(int x)
{
dfn[x]=low[x]=++id, sk[++top]=x, ins[x]=1;
for(int i=1; i<=n; ++i)
if(mp[x][i])
if(!dfn[i]) Tarjan(i),low[x]=std::min(low[x],low[i]);
else if(ins[i]) low[x]=std::min(low[x],dfn[i]);
else ;//
if(low[x]==dfn[x])
{
++cnt;
do{
bel[sk[top]]=cnt, ins[sk[top]]=0, scc[cnt].push_back(sk[top]);
}while(x!=sk[top--]);
}
}
inline bool cmp(const int &a,const int &b){
return dgr[a]<dgr[b];
}
void Insert(int x)
{
sk[++top]=x;
for(int i=nxt[x]; i/*necessary(对于单独一个点)*/&&i!=x; i=nxt[i]) sk[++top]=i;
}
int main()
{
n=read();
for(int i=2; i<=n; ++i)
for(int j=1; j<i; ++j)
mp[j][i]=read(), mp[i][j]=mp[j][i]^1;
for(int i=1; i<=n; ++i)
if(!dfn[i]) Tarjan(i);
for(int l,r,sz,i=1; i<=cnt; ++i)
{
l=r=scc[i][0], sz=scc[i].size();
for(int tmp,j=1; j<sz; ++j)//从1个点开始 扩展成哈密顿路径
{
tmp=scc[i][j];
if(mp[tmp][l]) nxt[tmp]=l, l=tmp;
else if(mp[r][tmp]) nxt[r]=tmp, r=tmp;
else
{
for(int k=l; nxt[k]; k=nxt[k])
if(mp[k][tmp]&&mp[tmp][nxt[k]])//在当前路径上找 连向tmp同时tmp连向其后边节点的
{
nxt[tmp]=nxt[k], nxt[k]=tmp;
break;
}
}
}
r=0;
for(int j=l; j; j=nxt[j])
if(r)
{
for(int k=r,las=l; ; las=k,k=nxt[k])
{
if(mp[j][k])
{
nxt[las]=nxt[l];
if(las!=l) nxt[l]=r;
r=k, l=j;
break;
}
if(k==l) break;
}
}
else if(mp[j][l]) r=l, l=j;
nxt[l]=r;
}
for(int i=1; i<=n; ++i)//对每个强连通分量拓扑排序
for(int j=1; j<=n; ++j)
if(bel[i]!=bel[j]&&mp[i][j]) ++dgr[bel[j]];
for(int i=1; i<=cnt; ++i) pos[i]=i, dgr[i]/=scc[i].size();
//度数要除以size 因为连向强连通分量i的点会连向i中所有点(竞赛图) 而此时拓扑是要对强连通分量的入度排序
std::sort(pos+1,pos+1+cnt,cmp);
for(int i=1; i<=n; ++i)
{
top=0, Insert(i);//直接加入i这个强连通分量即可,走一遍回路,然后再从终点走到下一个强连通分量
//bel[i]中每个点一定都可以走到下一个强连通分量j,否则bel[i]和j就成了环(同一个强连通分量)了
for(int j=1; j<=cnt; ++j)//按照拓扑序添加scc(路径)
if(dgr[pos[j]]>dgr[bel[i]])
Insert(scc[pos[j]][0]);
printf("%d ",top);
for(int i=1; i<top; ++i) printf("%d ",sk[i]);
printf("%d\n",sk[top]);
}
return 0;
}
BZOJ.4727.[POI2017]Turysta(哈密顿路径/回路 竞赛图)的更多相关文章
- BZOJ 4727: [POI2017]Turysta
4727: [POI2017]Turysta Time Limit: 20 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 117 Solved ...
- BZOJ4727 [POI2017]Turysta 【竞赛图哈密顿路径/回路】
题目链接 BZOJ4727 题解 前置芝士 1.竞赛图存在哈密顿路径 2.竞赛图存在哈密顿回路,当且仅当它是强联通的 所以我们将图缩点后,拓扑排序后一定是一条链,且之前的块内的点和之后块内的点的边一定 ...
- bzoj千题计划232:bzoj4727: [POI2017]Turysta
http://www.lydsy.com/JudgeOnline/problem.php?id=4727 竞赛图tarjan缩点后得到的拓扑图一定是一条链 因为竞赛图任意两点的前后顺序确定,只有一种拓 ...
- BZOJ 4726: [POI2017]Sabota?
4726: [POI2017]Sabota? Time Limit: 20 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 301 Solved ...
- BZOJ 4726: [POI2017]Sabota? 树形dp
4726: [POI2017]Sabota? 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4726 Description 某个公司有n ...
- bzoj 4725 [POI2017]Reprezentacje ró?nicowe 暴力
[POI2017]Reprezentacje ró?nicowe Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 141 Solved: 67[Sub ...
- bzoj 4724 [POI2017]Podzielno 二分+模拟
[POI2017]Podzielno Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 364 Solved: 160[Submit][Status][ ...
- bzoj 4723 [POI2017]Flappy Bird 模拟
[POI2017]Flappy Bird Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 482 Solved: 196[Submit][Status ...
- ACM学习历程—BZOJ 2115 Xor(dfs && 独立回路 && xor高斯消元)
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2115 题目大意是求一条从1到n的路径,使得路径xor和最大. 可以发现想枚举1到n的所有路 ...
随机推荐
- diff 命令用法--如何打补丁【原创--学习笔记】
diff 命令用法 1.”-u”:表示在比较结果中输出上下文中一些相同的行,这有利于人工定位 2.“-r“:表示递归比较各个子目录下的文件 3.“-N“:将不存在的文件当作空文件 4.“-w“:忽略对 ...
- java并发编程系列一、多线程
一.什么是线程 一个应用就是一个进程.一个进程由多个线程组成.一个生产车间比作是一个进程.工人比作是线程.当任务比较多的时候,增加工人可以提高效率,同时成本就是支付费用(机器资源,内存)也会增加. p ...
- 错误代码 1045 Access denied for user 'root'@'localhost' (using password:YES)
1 前言 现象是用MySQL 5.7 Command Line Client可以使用root账号进入,但是其它navicat,phpsqladmin,mysql workbench,heidisql用 ...
- 擅于使用JS的eval方法
样例如下: var appsDetails = {“app1”:"", “app2”:"", “app3”:"", “app4”:" ...
- 用java实现邮件发送验证码
java实现邮件发送验证码 建议不要用qq邮箱,我使用qq邮箱直接一直给我报530错误,我一直认为我代码写的有错误或者POP3/SMTP服务没弄好.所以建议注册个别的邮箱,我就申请了个网易163邮箱瞬 ...
- 配置_DruidDataSource参考配置
配置_DruidDataSource参考配置 <!-- 数据库驱动 --> <property name="driverClassName" value=&quo ...
- JMeter 聚合报告 90%响应时间
90%的响应时间理解 官方解释:90% Line - 90% of the samples took no more than this time. The remaining samples at ...
- 《剑指offer》-数据流中的中位数
如何得到一个数据流中的中位数?如果从数据流中读出奇数个数值,那么中位数就是所有数值排序之后位于中间的数值.如果从数据流中读出偶数个数值,那么中位数就是所有数值排序之后中间两个数的平均值. 最开始的思路 ...
- [转]一个研究生毕业以后的人生规划[ZZ]
只有选择去国内的大公司或外企才是出路 文章转载如下: 我今年39岁了, 25岁研究生毕业,工作14年,回头看看,应该说走了不少的弯路,有一些经验和教训.现在开一个小公司,赚的钱刚够养家糊口的.看看这些 ...
- python全栈开发day37-html
web准备总结: 结构标准:相当于人的身体.html就是用来制作网页的. 表现标准: 相当于人的衣服.css就是对网页进行美化的. 行为标准: 相当于人的动作.JS就是让网页动起来,具有生命力的 1. ...