CF-1562B- Scenes From a Memory

2022-11-10,

Problem - 1562B - Codeforces

题意:给定一个字符串,每次操作可以选择这个字符串中的一种字符,将他们全部都减1,最多K次操作,问可以形成的字典大小最小的字符串。

题解:首先我们观察这个字符串,很明显,如果两个字符不相同,我们把大的减小到和小的相等,然后再两个一起减小,就省去了一部分的操作,从而减少操作次数,因此,我们先把每种字符按照第一个先后出现的顺序存储,然后遍历这个数组,判断你当前读入的字符是不是到现在为止的最大字符,如果不是,就证明这个字符比前面的小,这样这个字符肯定就已经被连带减小了

例如:c b 在减小c的时候,当c等于了b,就可以同时减小两个。 所以当该字符不是最大字符时,就不需要管它。如果是,就判断该字符与上一个最大字符的差值,即你要变成上一个字符需要几次操作,然后与前面需要的操作相加,看看是否超过最大操作数,如果超过就能减少几个减少几个,没超过就加到总操作数里继续看下一个字符。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<ll,ll> pll;
const int N=4e4+5;
const ll mod=1e9+7;
ll dp[N],vis[26];
vector<ll> q[N],p;
signed main(){
ios::sync_with_stdio(false);
cin.tie(0);
ll t;cin>>t;
while(t--){
ll n,m;cin>>n>>m;
string s;cin>>s;
p.clear();
memset(vis,0,sizeof(vis));
for(ll i=0;i<=25;i++) q[i].clear();
for(ll i=0;i<s.size();i++){
q[s[i]-'a'].push_back(i);//记录该字符出现的位置
if(!vis[s[i]-'a']){
vis[s[i]-'a']=1;
p.push_back(s[i]-'a');//记录每种字符出现的先后顺序
}
}
ll sum=0;
ll maxn=-1;
ll last=0;
for(ll i=0;i<p.size();i++){
maxn=max(maxn,p[i]);
if(p[i]!=maxn) continue;//判断是否最大值
if(p[i]-last+sum>m){
for(ll j=0;j<q[p[i]].size();j++){
s[q[p[i]][j]]-=(m-sum);//将每个位置的字符能减少几减少几
}
for(ll j=p[i]-(m-sum);j<p[i];j++){
for(ll z=0;z<q[j].size();z++){
s[q[j][z]]=p[i]-(m-sum)+'a';//减少过程中,判断有没有比它小的字符被连带减小,有就更新
}
}
break;
}
for(ll z=0;z<=p[i];z++){
for(ll j=0;j<q[z].size();j++){
s[q[z][j]]='a';//如果操作数足够,就肯定会减少到 a
}
q[z].clear();//该字符减少完后就没有了,所以要清零
}
sum+=p[i]-last;
last=p[i];
}
cout<<s<<endl;
}
}

CF-1562B- Scenes From a Memory的相关教程结束。

《CF-1562B- Scenes From a Memory.doc》

下载本文的Word格式文档,以方便收藏与打印。