• 51Nod 1561 另一种括号序列


    题目链接

    分析:

    卡常数~~~好气啊~~~这是看脸的时代啊~~~

    $A$代表$($的数量,$B$代表$)$的数量...

    如果$($的数量多于$)$的数量,那么最有方案显然是添加$A-B$个$)$...反之亦然...

    因为这两种情况的处理方法差不多,所以我们假定$A>B$...

    因为是有循环位移操作的,所以我们我们证明一定存在一种添加方案可以使得循环位移之后在最后添加$A-B$个$)$并且字典序最小...

    证明如下:

    我们把$($设为$+1)$设为$-1$,那么合法的序列就是每个位置的前缀和都$>=0$,那么我们考虑找到最后一个坏点(也就是不合法的点),然后把这个位置到最后的字符全部移到前面,这样就满足了前缀没有坏点,然后对于后面的坏点我们递归进行...

    那么如何找到字典序最小的满足要求的原串排列,考虑是循环位移,所以我们倍增字符串,那么一个合法的排列就是某个后缀的长度$n$的前缀,因为要字典序最小,所以我们通过后缀数组来解决...我们找到每个后缀,判断是否合法,怎么判断自己YY吧,有益身心健康~~~

    代码:

    #include<algorithm>
    #include<iostream>
    #include<cstring>
    #include<cstdio>
    //by NeighThorn
    using namespace std;
    
    const int maxn=4000000+5;
    
    int n,len,gs[maxn],sa[maxn],wb[maxn],wv[maxn],ran[maxn],sum[maxn],pre[maxn],suf[maxn];
    
    char s[maxn],ans[maxn];
    
    inline bool cmp(int *x,int a,int b ,int l){
    	return x[a]==x[b]&&x[a+l]==x[b+l];
    }
    
    inline void da(int *sa,int *x,int n,int m){
    	int i,j,p,*y=wb;
    	for(i=0;i<m;i++) gs[i]=0;
    	for(i=0;i<n;i++) gs[x[i]]++;
    	for(i=1;i<m;i++) gs[i]+=gs[i-1];
    	for(i=n-1;~i;i--) sa[--gs[x[i]]]=i;
    	for(j=1,p=1;p<n;j<<=1,m=p){
    		for(i=n-j,p=0;i<n;i++) y[p++]=i;
    		for(i=0;i<n;i++) if(sa[i]>=j) y[p++]=sa[i]-j;
    		for(i=0;i<n;i++) wv[i]=x[y[i]];
    		for(i=0;i<m;i++) gs[i]=0;
    		for(i=0;i<n;i++) gs[wv[i]]++;
    		for(i=1;i<m;i++) gs[i]+=gs[i-1];
    		for(i=n-1;~i;i--) sa[--gs[wv[i]]]=y[i];
    		p=1;swap(x,y);x[sa[0]]=0;
    		for(i=1;i<n;i++) x[sa[i]]=cmp(y,sa[i],sa[i-1],j)?p-1:p++;
    	}
    }
    
    signed main(void){
    	int i,j;
    	scanf("%s",s);n=strlen(s);len=n<<1;
    	for(i=0;i<n;i++) ran[i]=(int)s[i]-39;
    	for(i=n;i<len;i++) ran[i]=(int)s[i-n]-39,s[i]=s[i-n];
    	da(sa,ran,len+1,5);
    	for(i=0;i<n;i++) sum[i]=sum[i-1]+(s[i]=='('?1:-1);
    	pre[0]=sum[0];
    	for(i=1;i<n;i++) pre[i]=pre[i-1]>sum[i]?sum[i]:pre[i-1];
    	suf[n-1]=sum[n-1];
    	for(i=n-2;i>=0;i--) suf[i]=suf[i+1]>sum[i]?sum[i]:suf[i+1];
    	for(i=1;i<=len;i++)
    		if(sa[i]<n){
    			int tmp,tmp1=suf[sa[i]]-sum[sa[i]-1],tmp2=sum[n-1]-sum[sa[i]-1]+pre[sa[i]-1];
    			tmp=tmp1>tmp2?tmp2:tmp1;
    			if(sum[n-1]>=0&&tmp>=0){
    				for(j=0;j<n;j++) ans[j]=s[sa[i]+j];
    				for(j=0;j<sum[n-1];j++) ans[j+n]=')';
    				puts(ans);break;
    			}
    			tmp-=sum[n-1];
    			if(sum[n-1]<0&&tmp>=0){
    				tmp=-sum[n-1];
    				for(j=0;j<tmp;j++) ans[j]='(';
    				for(j=0;j<n;j++) ans[j+tmp]=s[sa[i]+j];
    				puts(ans);break;
    			}
    		}
    	return 0;
    }
    

      


    By NeighThorn

  • 相关阅读:
    自定义异常
    this()和super()的区别
    Eclipse快速创建函数
    Eclipse的.java文件有蓝色三角符号
    可变数量参数
    近期的目标
    怎么在电脑右键的新建里面添加.md文件
    数组的排序(从小到大)
    Arrays类
    不用乘法计算乘法(快速计算)
  • 原文地址:https://www.cnblogs.com/neighthorn/p/6649996.html
Copyright © 2020-2023  润新知