bzoj 4278: [ONTAK2015]Tasowanie (后缀数组)

4278: [ONTAK2015]Tasowanie

Time Limit: 10 Sec   Memory Limit: 256 MB
Submit: 313   Solved: 140
[ Submit][ Status][ Discuss]

Description

给定两个数字串A和B,通过将A和B进行二路归并得到一个新的数字串T,请找到字典序最小的T。

Input

第一行包含一个正整数n(1<=n<=200000),表示A串的长度。
第二行包含n个正整数,其中第i个数表示A[i](1<=A[i]<=1000)。
第三行包含一个正整数m(1<=m<=200000),表示B串的长度。
第四行包含m个正整数,其中第i个数表示B[i](1<=B[i]<=1000)。

Output

输出一行,包含n+m个正整数,即字典序最小的T串。

Sample Input

6
1 2 3 1 2 4
7
1 2 2 1 3 4 3

Sample Output

1 1 2 2 1 2 3 1 2 3 4 3 4

HINT

Source

[ Submit][ Status][ Discuss]

题解:后缀数组

将两个串用一个分隔符连接起来,然后求rank数组.需要注意的是分隔符一定要加较大的,否则会出现2 0 2 1(0是分隔符)正确的答案是212,但是会选择出221.在整个串的最后也要加一个大的分隔符,否则会出现2 1 1003 2正确的答案还是212,但是如果不加分隔符会选择出221.

不过这题的数据较水,网上的很多程序都是有bug的,但是都过了。。。。表示还是应该加强一下数据。

将两个头指针放到两个串开始的位置,每次选取开头位置中rank较小的。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define N 5000007
using namespace std;
int n,m,a[N],v[N],xx[N],yy[N],*x,*y,sa[N],rank[N],p;
int cmp(int i,int j,int l) {
	return y[i]==y[j]&&(i+l>n?-1:y[i+l])==(j+l>n?-1:y[j+l]);
}
void get_sa()
{
	x=xx; y=yy;
	memset(v,0,sizeof(v));
	int m1=1005;
	for (int i=1;i<=n;i++) v[x[i]=a[i]]++;
	for (int i=1;i<=m1;i++) v[i]+=v[i-1];
	for (int i=n;i>=1;i--) sa[v[x[i]]--]=i;
	for (int k=1;k<=n;k<<=1) {
		p=0;
		for (int i=n-k+1;i<=n;i++) y[++p]=i;
		for (int i=1;i<=n;i++)
		 if (sa[i]>k) y[++p]=sa[i]-k;
		for (int i=1;i<=m1;i++) v[i]=0;
		for (int i=1;i<=n;i++) v[x[y[i]]]++;
		for (int i=1;i<=m1;i++) v[i]+=v[i-1];
		for (int i=n;i>=1;i--) sa[v[x[y[i]]]--]=y[i];
		swap(x,y); p=2; x[sa[1]]=1;
		for (int i=2;i<=n;i++) 
		 x[sa[i]]=cmp(sa[i],sa[i-1],k)?p-1:p++;
		if (p>n) break;
		m1=p+1;
	}
	for (int i=1;i<=n;i++) rank[sa[i]]=i;
}
int main()
{
	freopen("a.in","r",stdin);
	scanf("%d",&n);
	for (int i=1;i<=n;i++) scanf("%d",&a[i]);
	a[++n]=1003;
	scanf("%d",&m);
	for (int i=1;i<=m;i++) scanf("%d",&a[++n]);
	a[++n]=1003;
	get_sa();
	int l=1; int r=n-m;
//	cout<<l<<" "<<r<<endl;
    int cnt=0;
    for (int i=1;i<=n-2;i++){
    	if (rank[l]<rank[r]) printf("%d ",a[l]),l++;
		else printf("%d ",a[r]),r++;
		if (l==n-m-1) {
			for (int j=r;j<=n-1;j++) printf("%d ",a[j]);
			break;
		}
		if (r==n) {
			for (int j=l;j<=n-m-2;j++) printf("%d ",a[j]);
			break;
		}
	}
	printf("\n");
}



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值