zoukankan      html  css  js  c++  java
  • CF1053E Euler tour

    题意

    给出一个某些位置不全的欧拉序,求出一个符合条件的,或输出不行
    传送门
    (n le 5*10^5)

    思路

    终于不是一道神仙(dp)
    变成了一道神仙构造
    以下简称两相同数围成的是一个区间,基本性质:

    1. 两个相同数之间的长度为奇数
    2. 头尾一定相同
    3. 区间要么完全包含要么不相交
    4. 某段区间内已确定的落单的数一定小于区间(0)的个数(+1),因为一个落单的数除了叶子总得再配一个
    5. 对任意一棵大小为(x)的子树,序列长度为(2x-1)

    首先区间不相交就让我们可以分治搞下去,先递归到最小的区间处理,然后处理完此区间对答案就没有影响了,可以直接删掉(浓缩成一个根节点代替整个子树),具体就是维护一下前驱和后继循环时一段一段跳就好了。

    接着要考虑最小的区间怎么办了。考虑一下填完后确定的数一定是(lfloor frac{区间长度+1}2 floor),那么首先从前往后在多出来的空格填未出现过的数字。
    接着就开始了,形如(120)(021)的,只要填成(121),也就是我们要不停的找如(xy0)(0xy)的某三个数,并且填完后把它们当做只有(2)也就是父亲节点就好了。
    如果还有是没填的怎么办呢?发现当前区间的父亲可以用来分割两个区间。
    例如填完后变成了(1[20304]1)括号表示当前区间,无法用前面的方法再来填,那就可以用(1)来填补剩下的空格变为(1213141)

    #include <bits/stdc++.h>
    using std::deque;
    const int N=1000100;
    int suf[N],pre[N],Next[N],last[N],c[N],a[N],n,m,now;
    struct note{
    	int x,p;
    };
    deque <note> q; 
    void endd(){
    	puts("no");
    	exit(0);
    } 
    void del(int x,int y){
    	suf[x]=suf[y];
    	pre[suf[y]]=x;
    }
    int get(){
    	while (c[now]) now++;
    	if (now>n) endd();
    	c[now]=1;
    	return now;
    }
    void solve(int l,int r){
    	if (r<l) return;
    	if ((r-l)&1) endd();
    	for (int i=l;i<=r;i=suf[i]){
    		if (!a[i]) continue;
    		while (Next[i]){
    			int to=Next[i];
    			if (to>r) endd();
    			solve(suf[i],pre[to]);
    			del(i,to);
    			Next[i]=Next[to];
    		}
    	}
    	int sum=0,sum0=0;
    	for (int i=l;i<=r;i=suf[i]){
    		sum+=(a[i]>0);
    		sum0+=(!a[i]);
    	}
    	if (sum0<sum-1) endd();
    	for (int i=l;i<=r && sum0>sum;i=suf[i])
    		if (!a[i]) a[i]=get(),sum0--,sum++;
    	int rt=a[pre[l]];
    	for (int i=l;suf[i]<=r;i=suf[i]){
    		while (i>l && suf[i]<=r && a[i] && a[suf[i]] && (!a[pre[i]])){
    			a[pre[i]]=a[suf[i]];
    			del(pre[i],suf[i]);
    			i=pre[pre[i]];
    		}
    		while (i>=l && suf[suf[i]]<=r && a[i] && a[suf[i]] && (!a[suf[suf[i]]])){
    			a[suf[suf[i]]]=a[i];
    			del(i,suf[suf[i]]);
    			i=pre[i];
    		}
    	} 
    	for (int i=l;i<=r;i=suf[i])
    		if (!a[i]) a[i]=rt;
    }
    int main(){
    	scanf("%d",&n);
    	m=2*n-1;
    	for (int i=1;i<=m;i++) scanf("%d",&a[i]),c[a[i]]++;
    	if (a[1]!=a[2*n-1] && a[1] && a[2*n-1]) endd();
    	a[1]=a[m]=(a[1]|a[m]);
    	for (int i=m;i>=1;i--)
    		Next[i]=last[a[i]],last[a[i]]=i;
    	now=1;
    	suf[0]=1; 
    	for (int i=1;i<=m;i++) pre[i]=i-1,suf[i]=i+1;
    	solve(1,m);
    	puts("yes");
    	for (int i=1;i<=m;i++) printf("%d ",a[i]);
    } 
    

    后记

    神仙啊,其实我还是不怎么懂

  • 相关阅读:
    【noip2014】解方程
    2013南大专业课试题答案
    如何计算多边形面积
    坐标转换成点
    几何对象和空间参考
    StoryBoard(动画故事版)
    五种画刷
    SL的六种变换
    SL中MediaElement(视频播放)
    SL的四种样式
  • 原文地址:https://www.cnblogs.com/flyfeather6/p/11798090.html
Copyright © 2011-2022 走看看