详细做法以及证明请看论文《Hamiltonian paths in the square of a tree》

首先将1到n的路径提取出来,作为主干。

定义毛毛虫为去掉叶子之后只有一条单链的树,定义non-trivial的毛毛虫为单链非空的毛毛虫。

对于主干上每个点,计算它的非主干部分是否是毛毛虫,如果某个部分不是毛毛虫,那么肯定无解。

将主干上每个点划分为两类:

A:non-trivial的毛毛虫不超过1个。

B:non-trivial的毛毛虫恰有两个。

同时定义一个点是free的当且仅当它是单点。

如下图:

判断一下是否每对相邻的B类点中间都有free点,且所有B类点前后都有free点,如果没有,那么也无解。

现在必然存在一条以1为起点,且依次遍历完主干上每个点及其子树的2H-哈密顿路径。

对于1,如果它是free的,那么方案显然,否则从1出去的点必定是它子树里离它距离只有1的点。

然后一个一个构造,如果上一个点是第二层的点,那么这一部分起点必须为主干上的点,否则起点选第二层的点显然更优。

对于一个毛毛虫,可以直接构造出2H-哈密顿环,从中断开某条路径得到合法的方案。

如下图:

但是如果一个点是B类点,这个时候是不能如此构造的,只能是起点和终点都是第二层的点,而且起点优先考虑叶子节点,这个时候就又可以按之前的方法构造。

最后再检验一下终点是否为n即可,时间复杂度$O(n)$。

#include<cstdio>
#include<cstdlib>
#define N 500010
int n,i,j,x,y,S,T,d[N],g[N],v[N<<1],nxt[N<<1],ed,pre[N],sub[N];
int f[N],G[N],V[N],NXT[N];
int m,a[N],ans,fin[N],cnt,q[N],b[N<<1],tot;bool vip[N],fr[N];
inline void read(int&a){char c;while(!(((c=getchar())>='0')&&(c<='9')));a=c-'0';while(((c=getchar())>='0')&&(c<='9'))(a*=10)+=c-'0';}
void NO(){puts("BRAK");std::exit(0);}
inline void swap(int&a,int&b){int c=a;a=b;b=c;}
inline void reserve(int*a,int n){for(int i=1,j=n;i<j;i++,j--)swap(a[i],a[j]);}
inline void add(int x,int y){d[x]++;v[++ed]=y;nxt[ed]=g[x];g[x]=ed;}
inline void ADD(int x,int y){f[y]=x;V[++ed]=y;NXT[ed]=G[x];G[x]=ed;}
void dfs(int x,int y){
pre[x]=y;
for(int i=g[x];i;i=nxt[i])if(v[i]!=y)dfs(v[i],x);
}
bool spine(int x,int y){
int t=0;
for(int i=g[x];i;i=nxt[i])if(v[i]!=y&&d[v[i]]>1){
if(!spine(v[i],x))return 0;
if(t++)return 0;
}
return 1;
}
inline int any_secondary_node(int x){
for(int i=g[x];i;i=nxt[i])if(!vip[v[i]])return v[i];
}
inline int a_secondary_node_preferably_leaf(int x){
for(int i=g[x];i;i=nxt[i])if(!vip[v[i]]&&d[v[i]]==1)return v[i];
return any_secondary_node(x);
}
inline int another_secondary_node(int x,int y){
for(int i=g[x];i;i=nxt[i])if(!vip[v[i]]&&v[i]!=y)return v[i];
}
void findspine(int x,int y){
q[++cnt]=x;
for(int i=g[x];i;i=nxt[i])if(v[i]!=y)if(d[v[i]]==1)ADD(x,v[i]);else findspine(v[i],x);
}
inline int left(int x){return x==1?tot:x-1;}
inline int right(int x){return x==tot?1:x+1;}
inline void extend(int x,int a=0,int b=0){
for(int i=G[x];i;i=NXT[i])if(V[i]!=a&&V[i]!=b)fin[++ans]=V[i];
}
inline void path_caterpillar(int x,int S,int T){
fin[++ans]=S;
if(S==T)return;
int i,j,A,B;
q[cnt=1]=x;
for(i=g[x];i;i=nxt[i])if(!vip[v[i]]){
if(d[v[i]]==1)ADD(x,v[i]);
else{
if(cnt>1)reserve(q,cnt);
findspine(v[i],x);
}
}
for(i=1;i<=cnt;i++){
b[i]=i&1?q[i]:-q[i];
b[cnt+cnt-i+1]=-b[i];
}
for(tot=0,i=1;i<=cnt+cnt;i++){
if(b[i]<0)if(!G[-b[i]])continue;
b[++tot]=b[i];
}
if(d[S]>1)A=S;else A=-f[S];
if(d[T]>1)B=T;else B=-f[T];
for(i=1;i<=tot;i++)if(b[i]==A)break;
if(A<0)extend(-A,S,T);
if(b[left(i)]==B)
for(j=right(i);b[j]!=B;j=right(j))if(b[j]>0)fin[++ans]=b[j];else extend(-b[j]);
else
for(j=left(i);b[j]!=B;j=left(j))if(b[j]>0)fin[++ans]=b[j];else extend(-b[j]);
if(B<0&&A!=B)extend(-B,S,T);
fin[++ans]=T;
}
int main(){
read(n);
for(i=1;i<n;i++)read(x),read(y),add(x,y),add(y,x);
dfs(n,ed=0);
for(i=1;i!=n;i=pre[i])a[++m]=i;a[++m]=i;
for(i=1;i<=m;i++)vip[a[i]]=1;
for(i=1;i<=m;i++){
x=a[i];fr[x]=1;
for(j=g[x];j;j=nxt[j]){
y=v[j];
if(vip[y])continue;
fr[x]=0;
if(d[y]==1)continue;
sub[x]++;
if(sub[x]>2)NO();
if(!spine(y,x))NO();
}
}
for(i=1,j=0;i<=m;i++){
x=a[i];
if(fr[x])j=1;
else if(sub[x]==2){
if(j!=1)NO();
j=2;
}
}
if(j==2)NO();
S=a[1];
if(fr[a[1]])T=a[1];else T=any_secondary_node(a[1]);
path_caterpillar(a[1],S,T);
for(i=2;i<=m;i++){
x=a[i];
if(fr[x])S=T=x;
else if(!vip[T]){
S=x;
T=a_secondary_node_preferably_leaf(x);
}else{
S=a_secondary_node_preferably_leaf(x);
T=sub[x]<2?x:another_secondary_node(x,S);
}
path_caterpillar(x,S,T);
}
if(fin[n]!=n)NO();
for(i=1;i<=n;i++)printf("%d\n",fin[i]);
return 0;
}

  

BZOJ3424 : Poi2013 Multidrink的更多相关文章

  1. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  2. [POI2013]Łuk triumfalny

    [POI2013]Łuk triumfalny 题目大意: 一棵\(n(n\le3\times10^5)\)个结点的树,一开始\(1\)号结点为黑色.\(A\)与\(B\)进行游戏,每次\(B\)能选 ...

  3. [POI2013]Polaryzacja

    [POI2013]Polaryzacja 题目大意: 给定一棵\(n(n\le250000)\)个点的树,可以对每条边定向成一个有向图,这张有向图的可达点对数为树上有路径从\(u\)到达\(v\)的点 ...

  4. [POI2013]Taksówki

    [POI2013]Taksówki 题目大意: ABC三地在同一条直线上,AC相距\(m(m\le10^{18})\)米,AB相距\(d\),B在AC之间.总共有\(n(n\le5\times10^5 ...

  5. [POI2013]Usuwanka

    [POI2013]Usuwanka 题目大意: 一排\(n\)个球,有黑白两种颜色.每取走一个球会在原位置放一个水晶球.求构造一种取球方案,满足: 每次取走\(k\)个白球和\(1\)个黑球: 一次取 ...

  6. [POI2013]Morskie opowieści

    [POI2013]Morskie opowieści 题目大意: 一个\(n(n\le5000)\)点\(m(m\le5000)\)边无向图,边权均为\(1\),有\(k(k\le10^6)\)个询问 ...

  7. [POI2013]Bajtokomputer

    [POI2013]Bajtokomputer 题目大意: 给定一个长度为\(n(n\le10^6)\)的由\(\{-1,0,1\}\)组成的序列,你可以进行\(A_i+=A_{i-1}\)这样的操作, ...

  8. POI2013题解

    POI2013题解 只做了BZ上有的\(13\)道题. 就这样还扔了两道神仙构造和一道计算几何题.所以只剩下十道题了. [BZOJ3414][Poi2013]Inspector 肯定是先二分答案,然后 ...

  9. 【BZOJ3416】Poi2013 Take-out 栈

    [BZOJ3416]Poi2013 Take-out Description 小F喜欢玩一个消除游戏——take-out 保证k+1|n,保证输入数据有解这是一个单人游戏 游戏者的目标是消除初始时给定 ...

随机推荐

  1. Java大数处理类:BigInteger类和BigDecimal类

    当我们要处理非常大的数据时,平常用的数据类型已不足以表示,在Java中有两个类BigInteger和BigDecimal分别表示大整数类和大浮点数类,这两个类在理论上只要计算机内存足够大就能够表示无线 ...

  2. 报错 - Command /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/lex failed with exit code 1

    这里有两种情况:第一种是某xxx.m文件不存在或路径不对,而且里面有5.1什么的解决方法如下:在Build Phases-->Complie Sources中找到有两个xxx.m文件,一个正常, ...

  3. 按键的使用(一)------verilog

    按键在项目中应用还是很频繁的,这里主要介绍按键的几种用法. 1.按下一次有效:按下一次计数器增加一下. 2.按下连续有效:按下不松,计数器就一直增加. 3.按下无效,松开有效:按下时计数器值不变,按键 ...

  4. yum -y list java* 查看当前java的版本

    [root@NB ok]# yum -y list java* Loaded plugins: fastestmirror, refresh-packagekit, security Loading ...

  5. C#回顾 – 4.IEnumerable 集合

         

  6. 解析PHP处理换行符的问题 \r\n

    一首先说说 \r 与\n的区别回车”(Carriage Return)和“换行”(Line Feed)这两个概念的来历和区别.在计算机还没有出现之前,有一种叫做电传打字机(Teletype Model ...

  7. 【Jquery】【控件】flexigrid 自定义查询

    最近用flexigrid作报表,需要自定义条件进行查询,界面如下: 翻了半天文档也没找到如何用POST方法自行传递参数进行查询. 找了一个台湾人写的博客,却要改flexigrid的源代码,更不靠谱. ...

  8. html5 Canvas绘制图形入门详解

    html5,这个应该就不需要多作介绍了,只要是开发人员应该都不会陌生.html5是「新兴」的网页技术标准,目前,除IE8及其以下版本的IE浏览器之外,几乎所有主流浏览器(FireFox.Chrome. ...

  9. C# IP地址与整数之间的转换

    IP地址与整数之间的转换 1.IP地址转换为整数 原理:IP地址每段可以看成是8位无符号整数即0-255,把每段拆分成一个二进制形式组合起来,然后把这个二进制数转变成一个无符号的32位整数. 举例:一 ...

  10. linux环境下libevent的使用

    step1:安装libevent yum install libevent step2: 代码入下: #include <sys/socket.h> #include <sys/ty ...