【贪心+一点小思路】Zoj - 3829 Known Notation

时间:2023-03-09 19:25:30
【贪心+一点小思路】Zoj - 3829 Known Notation

借用别人一句话,还以为是个高贵的dp... ...

一打眼一看是波兰式的题,有点懵还以为要用后缀表达式或者dp以下什么什么的,比赛后半阶段才开始仔细研究这题发现贪心就能搞,奈何读错题了!!交换的时候可以任意两个字符交换然而就那么看成了只能相邻的数字字符与'*'字符交换....../(ㄒoㄒ)/~~...但赛后补题的时候发现细节考虑的不好,还是应该锻炼下自己的逻辑整理...

怎么贪咧...

第一步,全数字串即合法,直接输出0即可;

第二步,数字不够的话要添。所谓数字够,即数字的个数至少要比星号个数多1,否则可以把缺少的数字(al+1-di)加在最开头,这样是最直接理想的。相应的,这也相当于增添操作的次数进行了(al+1-di)次;

第三步,在数字个数与星号个数都合法的条件下,不合法星号的位置要进行交换,交换策略是和该星号后的一个数字交换;这样的话就能保证每一个星号前是合法的,可能会有人对这一步不太理解:

 for(int i = ; i < len; i++)
{
if(str[i] == '*') ta++;
else td++;
if(td-ta<)
{
td++; ta--; ans++;
}
}

因为所有数字个数与字符个数都已经是合法的了,所以大可以不必考虑交换的具体位置,保证星号前合法的交换总会合法的。

附代码:

 #include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<string>
using namespace std;
int main()
{
int T; cin >> T;
while(T--)
{
string str;
cin >> str;
int len = str.length(); int al, di; al = di = ; //初始串的数字个数与星号个数
for(int i = ; i < len; i++) if(str[i] == '*') al++;
if(al == ) {printf("0\n"); continue;}
di = len-al;
int need; //是否缺少数字字符
if(di - al >= ) need = ; else need = al+-di; int td, ta; td = need; ta = ;
int ans = need;
for(int i = ; i < len; i++)
{
if(str[i] == '*') ta++;
else td++;
if(td-ta<)
{
td++; ta--; ans++;
}
}
cout << ans << endl;
}
return ;
}