设删掉第(i)个字符之后得到的字符串为(s_i) , 请按照字典序对(s_1,s_2,……,s_n)从小到大排序。若两个字符串相等,则认为编号小的字符串字典序更小。
考虑删掉第(i)位和第(j)位,其首尾相等,所以只要考虑(s[i+1,j])和(s[i,j-1])的大小就行了。
考虑写这样一个(cmp)函数,它的格式已经是固定了的,那么关键是需要预处理出某一段区间是相等的。
设(f[i])表示(i)位置之后,第一个(s[i]!=s[i+1])的位置
(if(f[i]>=j)) 中间串相等,编号小的在前
(if(f[i]<j)) 比较这第一个不相同的位置的大小
详见代码
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define debug(...) fprintf(stderr,__VA_ARGS__)
#define Debug(x) cout<<#x<<"="<<x<<endl
#define rank Rank
using namespace std;
typedef long long LL;
const int INF=1e9+7;
inline LL read(){
register LL x=0,f=1;register char c=getchar();
while(c<48||c>57){if(c=='-')f=-1;c=getchar();}
while(c>=48&&c<=57)x=(x<<3)+(x<<1)+(c&15),c=getchar();
return f*x;
}
const int N=1e6+5;
int same[N],rank[N],n;
char s[N];
inline bool cmp(int a,int b){
if(a<b){
if(same[a]>=b) return 1;
return s[same[a]+1]<s[same[a]];
}
else return !cmp(b,a);//我觉得这样比较好,就固定了格式只有一种
/*else{
if(same[b]>=a) return 0;
return s[same[b]]<s[same[b]+1];
}*/
}
int main(){
n=read();scanf("%s",(s+1));
for(int i=1,t=1;i<=n;i++){
if(t<i) t=i;
while(t<n&&s[t]==s[t+1]) t++;
same[i]=t;
}
for(int i=1;i<=n;i++) rank[i]=i;
sort(rank+1,rank+n+1,cmp);
for(int i=1;i<=n;i++) printf("%d ",rank[i]);
}