博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【AtCoder】ARC 081 E - Don't Be a Subsequence
阅读量:6592 次
发布时间:2019-06-24

本文共 1265 字,大约阅读时间需要 4 分钟。

【题意】给定长度为n(<=2*10^5)的字符串,求最短的字典序最小的非子序列字符串。

【算法】字符串DP

【题解】

先考虑计算最短长度,再考虑求字典序最小。

关键在于发掘出【最短的非子序列字符串】具有最优子结构,定义f(s)为字符串s的最短的非子序列字符串长度,假设最短的非子序列字符串为t,当t的第一个字母是c(任意字母)时,只有两种情况:

①s中无c,f(s)=1最短。

②对于s中最左边位置p的c,f(s)=f(s.suffix(p+1))+1,最后这个+1就是c。

很熟悉对吗?①是终止条件,②是状态转移,满足最优子结构性质,当确定第一个字母后,剩余部分可以转化为计算完毕的子问题

那么正式定义状态转移方程,令f[i]表示字符串的后缀i的最短非子序列字符串的长度,pos[i][j]表示从位置i开始第一个字母j出现的位置。

状态转移方程:f[i]=min(f[pos[i][j]+1])+1,0<=j<26。

最后要求字典序最小,从f[0]开始对于每一步找到最小的字母c满足f[i]==f[pos[i][c]]+1输出即可。

复杂度O(n*26)。

考虑清楚边界问题!

#include
#include
#include
using namespace std;const int maxn=200010;int f[maxn],pos[maxn][30],n;char s[maxn];int main(){ scanf("%s",s); n=strlen(s); for(int i=0;i<26;i++)pos[n][i]=n; f[n]=1;f[n+1]=0;//即使没有后缀仍然需要长度为1,f[n]=1;如果没有相同字符的话就f[n+1]+1,所以f[n+1]=0。 for(int i=n-1;i>=0;i--){ for(int j=0;j<26;j++)pos[i][j]=pos[i+1][j]; pos[i][s[i]-'a']=i; f[i]=n+1; for(int j=0;j<26;j++)f[i]=min(f[i],f[pos[i][j]+1]+1); } int T=f[0],p=0; while(T--){ for(int j=0;j<26;j++)if(f[pos[p][j]+1]+1==f[p]){ putchar('a'+j); p=pos[p][j]+1; break; } } return 0;}
View Code

 

转载于:https://www.cnblogs.com/onioncyc/p/7418332.html

你可能感兴趣的文章
C/C++获取文件大小
查看>>
深入理解Java内存模型(五)——锁
查看>>
Chalubo僵尸网络来袭 IOT设备或将受到DDoS攻击
查看>>
如何实现百万TPS?详解JMQ4的存储设计
查看>>
这么说吧,NIO很简单,其实就是个牛逼IO
查看>>
使用Python快速获取公众号文章定制电子书(二)
查看>>
iOS下JS与OC互相调用(七)--Cordova 基础
查看>>
Three.js 关于立方体贴图产生边缘锯齿问题
查看>>
Nacos v0.7.0:对接CMDB,实现基于标签的服务发现能力
查看>>
【开发问题记录①】关于滑动CollectionView时ContentSize变化的问题
查看>>
java中GC的基本概念
查看>>
building xxx gradle project info的解决办法
查看>>
在 CentOS 7 上搭建 Jenkins + Maven + Git 持续集成环境
查看>>
数据结构与算法 | Leetcode 19. Remove Nth Node From End of List
查看>>
[LeetCode] 862. Shortest Subarray with Sum at Least K
查看>>
【分享】终端命令工具 自动生成vue组件文件以及修改router.js
查看>>
[LeetCode] Student Attendance Record I
查看>>
PHP回顾之多进程编程
查看>>
spring boot + redis
查看>>
Ajax技术细节
查看>>