链接:https://ac.nowcoder.com/acm/contest/551/D
来源:牛客网
时间限制:C/C++ 1秒,其他语言2秒
空间限制:C/C++ 524288K,其他语言1048576K
64bit IO Format: %lld
题目描述
CSL 以前不会字符串算法,经过一年的训练,他还是不会……于是他打算向你求助。
给定一个字符串,只含有可打印字符,通过删除若干字符得到新字符串,新字符串必须满足两个条件:
- 原字符串中出现的字符,新字符串也必须包含。
- 新字符串中所有的字符均不相同。
- 新字符串的字典序是满足上面两个条件的最小的字符串。
输入描述:
仅一行,有一个只含有可打印字符的字符串 s。
|s|≤105|s|≤105
输出描述:
在一行输出字典序最小的新字符串。
示例1
输入
复制
bab
输出
复制
ab
示例2
输入
复制
baca
输出
复制
bac
备注:
ASCII字符集包含 94 个可打印字符(0x21 - 0x7E),不包含空格。
地址:https://ac.nowcoder.com/acm/contest/551/D
思路:单调栈
先预处理出每个字符的最后出现位置,在维护一个单调栈,遍历字符串str,当str[i]没有入栈时,
若str[i]>栈顶元素,则直接入栈,
否则判断栈顶元素是否为最后一个此元素,若不是则出栈,最后再将str[i]入栈即可
Code :
#include<iostream>
using namespace std;
const int MAX_M=155;
int n;
string str;
int d[MAX_M],R[MAX_M];
bool boo[MAX_M];
int main()
{
ios::sync_with_stdio(false);
cin>>str;
n=str.length();
for(int i=0;i<n;++i)
R[str[i]]=i+1;
int t=0,m=0;
for(int i=0;i<MAX_M;++i)
if(R[i]) ++m;
for(int i=0;i<n&&t<m;++i)
if(!boo[str[i]]){
while(t>0&&str[i]<d[t]&&R[d[t]]>i+1){
boo[d[t]]=false; --t;
}
d[++t]=str[i]; boo[str[i]]=true;
}
for(int i=1;i<=m;++i)
cout<<char(d[i]);
cout<<endl;
return 0;
}