名称:字符串

来源:2019年陕西省选

链接

博客链接

题目链接

题目内容

题目描述

给出一个长度为\(n\)的由小写字母组成的字符串\(a\),设其中第\(i\)个字符为\(a_i(1≤i≤n)\)。

设删掉第\(i\)个字符之后得到的字符串为\(s_i\),请按照字典序对\(s_1,s_2,……,s_n\)从小到大排序。若两个字符串相等,则认为编号小的字符串字典序更小。

格式

输入

第一行一个整数\(n\)。

第二行一个长为\(n\)的由小写字母组成的字符串\(a\)。

输出

输出一行\(n\)个整数\(k_1,k_2,……,k_n\),用空格隔开。表示$ s_{k_1}<s_{k_2}<……<s_{k_n} $。

数据

样例

输入

7
aabaaab

输出

3 7 4 5 6 1 2

数据范围

对于所有数据, \(1\leq n\leq10^6\) 。

对于10%的数据, \(1\leq n\leq2000\) 。

对于另外20%的数据,\(1\leq n\leq10^5\) 。且任意两个相邻字符\(a_i,a_{i+1}\)不相等;

对于另外30%的数据,\(1\leq n\leq10^5\)。

对于余下40%的数据,无特殊限制。

提示

本题不需要SA或者SAM等高级算法。

题解

10分

暴力构造出\(s\),再用快速排序进行排序。时间复杂度为\(O(n^2log(n))\),在\(n\leq2000\)的数据下跑得过。

30分

注意到:其中20%的数据没相邻两个字符不相等。

引理

当其任意两个字符不相等时,\(s_i\)和\(s_j(i<j)\)的大小关系实际上就是\(a_{i+1}\)与\(a_i\)的大小关系。

证明:由题意得。

\[s_i[1……i-1]=a[1……i-1]=s_j[1……i-1]\\
s_i[j……n-1]=a[j+1……n]=s_j[j……n-1]
\]

重点在于比较\(s_i[i……j-1]\)和\(s_j[i……j-1]\)的大小关系。

\(s_i[i……j-1]\)正对应\(a[i+1……j]\);

\(s_j[i……j-1]\)正对应\(a[i……j-1]\);

而\(a[i+1]!=a[i]\),故二者的大小关系可以确定。

于是,我们可以开一个双端队列。逆序处理整个字符串。

当我们发现\(a[i+1]<a[i]\)时,则说明\(s_i\)比后面的(即已经被处理过放进双端队列里的)都要小,就把数字\(i\)放在双端队列的前面;否则说明\(s_i\)比后面的都要大,就把它放在双端队列的后面。

graph TD
a(开始处理状态i)-->b{a_i+1< a_i}
b-->|yes|c[将i插在双端队列前面]
b-->|no|d[将i插在双端队列后面]
c-->e(进入下一轮操作)
d-->e

最后我们按顺序将双端队列每一个位置上的数字。

100分

那么,我们如何拿到100分呢???

实际上,我们只需要将原来30分的做法进行扩展,或者说将所有情况转换为两两相邻字符不相等的情况就行。

引理

当\(a[i]=a[i+1]\)时,\(s[i]=s[i+1]\)

证明略

由此,我们就可以将字符串连续相同的一段进行压缩,其中每一个字母都代表着原字符串的一段区间。

例如\("bbbcaa"\)压缩成\("bca"\),并且处理出来如下数据:

字母 开始位置 结束位置
b 1 3
c 4 4
a 5 6

将压缩后的串排序得\(3\quad1\quad2\)。

将原来处理出来的数据带入得\((5\quad6)(1\quad2\quad3)(4)\)。

//C++
#include<iostream>
#include<stdio.h>
#include<string>
#include<deque>
using namespace std;
const int nn=1000001;
inline void output(long long o);
int start[nn],final[nn];
inline long long input();
string a;
deque<int>k;
int main()
{
int n=input(),size=0;
cin>>a;
for(int i=0,PREV=0;i<n;i++)
{
PREV=i;
while(a[i]==a[i+1]&&i<n)i++;
a[size]=a[i],start[size]=PREV+1,final[size++]=i+1;
}
for(int i=size-1;i>=0;i--)
if(a[i+1]<a[i])k.push_front(i);
else k.push_back(i);
for(;k.front()!=k.back();k.pop_front())
for(int i=start[k.front()],f=final[k.front()];i<=f;i++)output(i),putchar(' ');
for(int i=start[k.front()],f=final[k.front()];i<f;i++)output(i),putchar(' ');
output(final[k.front()]),putchar('\n');
return 0;
}
inline void output(long long o)
{
if(o<0)putchar('-'),o=-o;
if(o>=10)output(o/10);
putchar(o%10^'0');
}
inline long long input()
{
bool positive=true;
char now=getchar();
long long i=0;
for(;!isdigit(now);now=getchar())
if(now=='-')positive=!positive;
for(;isdigit(now);now=getchar())i=(i<<3)+(i<<1)+(now^'0');
return positive?i:-i;
}

[SNOI2019]字符串的更多相关文章

  1. 【LOJ#3095】[SNOI2019]字符串(后缀数组)

    [LOJ#3095][SNOI2019]字符串(后缀数组) 题面 LOJ 题解 首先画图看看如何比较两个串的大小,发现这个东西等价于求两个相邻的后缀的\(LCP\). 一个做法是求出\(SA\),然后 ...

  2. [洛谷P5329][SNOI2019]字符串

    题目大意:给一个长度为$n$的字符串$s$,字符串$p_i$为字符串$s$去掉第$i$个字符后形成的字符串.请给所有字符串$p_i$排序(相同字符串按编号排序) 题解:先去掉所有连续相同字符,因为它们 ...

  3. luogu P5329 [SNOI2019]字符串

    传送门 显然要写一个排序,那只要考虑cmp函数怎么写就行了.第\(i\)个字符串和第 \(j\)个,首先前\(min(i,j)-1\)个字符是相同的,然后就是要比较后缀\(min(i,j)\)和\(m ...

  4. 洛谷$P5329\ [SNOI2019]$字符串 字符串

    正解:字符串 解题报告: 传送门$QwQ$ 有两个很妙的方法,分别港下$QwQ$ 首先为了表示方便,这里和题面一样设$s_i$表示去掉第$i$个字母得到的字符串.另设$lcp(i,j)$表示$suf_ ...

  5. 题解 P5329 【[SNOI2019]字符串】

    用栈的做法来水一发. 首先我们有一个暴力的做法,枚举每个被删除的字符,然后排序输出,时间复杂度:$ O ( N \times N \times LogN ) $ . 然后我们观察一下数据,发现有一个数 ...

  6. 【LOJ】#3095. 「SNOI2019」字符串

    LOJ#3095. 「SNOI2019」字符串 如果两个串\(i,j\)比较\(i < j\),如果离\(a_{i}\)最近的不同的数是\(a_{k}\),如果\(j < k\)那么\(i ...

  7. 「SNOI2019」字符串

    题目 看起来非常一眼啊,我们完全可以\(std::sort\)来解决这歌问题 于是现在的问题转化成了比较函数怎么写 随便画一下就会发现前面的好几位是一样的,后面的好几位也是一样,只需要比较中间的一段子 ...

  8. [日常] SNOI2019场外VP记

    SNOI2019场外VP记 教练突然说要考一场别省省选来测试水平...正好还没看题那就当VP咯w... Day 1 八点开题打 .vimrc. 先看了看题目名...一股莫名鬼畜感袭来... 怎么T1就 ...

  9. Python高手之路【六】python基础之字符串格式化

    Python的字符串格式化有两种方式: 百分号方式.format方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存.[PEP-3101] This ...

随机推荐

  1. 仿LookUpEdit多列模糊搜索,功能比GridLookUpEdit强大,比SearhLookUpEdit方便

    先上效果图: 控件调用示例:(devexpress使用了16.2.6.0版本,可以根据实际需要进行版本转换) using System; using System.Collections.Generi ...

  2. Djangou中使用cookie和session

    一.会话跟踪 我们先需要了解是什么是会话!可以把会话理解为客户端与服务器之间的一次会话,在一次会话中可能会包含多次请求和响应,例如你给10086打个电话,你就是客户端,而10086服务人员就是服务器, ...

  3. BME200加密网关,在电力与工业应用的加密网关设计与介绍

    加密通信网关,顾名思义就是带加密的通信网关终端,  一般业内主是需用到是工业通信关行业的为主的.,BME200加密通信网关,主要电力和工业互联网相关领域开发的一款加密通信网关. 为什么出现加密网关 1 ...

  4. 机房ping监控 smokeping+prometheus+grafana(续) 自动获取各省省会可用IP

    一.前言 1.之前的文章中介绍了如何使用smokeping监控全国各省的网络情况:https://www.cnblogs.com/MrVolleyball/p/10062231.html 2.由于之前 ...

  5. Svn提交冲突问题

    MEclipse中的svn冲突解决办法: 1.        点击提交,报错——‘SVN提交’has encountered a problem. 2.        选中无法提交的文件,点击更新操作 ...

  6. x32下PsSetLoadImageNotifyRoutine的逆向

    一丶简介 纯属兴趣爱好.特来逆向玩玩. PsSetLoadImageNotifyRoutine 是内核中用来监控模块加载.操作系统给我们提供的回调. 我们只需要填写对应的回调函数原型即可进行加监控. ...

  7. Linux基础文件类型

    一.文件时间 ls -l 文件名 仅看的是文件的修改时间 [root@linux ~]# ls -l /etc/passwd -rw-r--r-- root root 5月 : /etc/passwd ...

  8. Linux基础文件打包

    一.打包与解压 (一).打包压缩 [root@linux ~]# tar -czf etc1.tar.gz /etc //-z 调用gzip [root@linux ~]# tar -cjf etc2 ...

  9. Netty服务端启动过程相关源码分析

    1.Netty 是怎么创建服务端Channel的呢? 我们在使用ServerBootstrap.bind(端口)方法时,最终调用其父类AbstractBootstrap中的doBind方法,相关源码如 ...

  10. stm8s和stm8l低功耗对比

    在低功耗应用中,一般来说mcu是常态halt模式,然后偶尔被唤醒(外部中断或者内部定时唤醒)进入运行模式.所以对比低功耗性能,一般来说只需要对比run模式和halt下的功耗即可,因为项目选用的是通过内 ...