您的位置:首页 > 其它

Cf 98 Div.2

2012-02-16 16:54 411 查看
又ak了一场 div2。。。

A题

给定一个CCPPCCPPCCCCCPPPP这样的只有C,P的串,从左到右,每次最多取5个字符,每次取的字符必须相同,问最少要取多少次。

我的代码比较短:

每次循环定个初始位置,以该位置的下一个位置开始for循环4次.

int main()
{
int ans,i,n;
char ch[110];
while(cin>>ch){
n= strlen(ch);
ans=0;
int st=0;
while(st<n){
for(i=st+1;i<st+5 && i<n;++i){
if( ch[i]!=ch[i-1] )
break;
}
ans++;
st= i;
}
cout<< ans <<endl;
}
}


B题

给定一个序列n<= 5000,每个元素大小介于1到n之间,问至少需要改变多少个元素才能使序列中的元素是两两不同即1到n每个数字各出现一次。

超水,统计没有出现的数字的个数。

C题

给定最多100000个区间,保证坐标x,y都是没有重复的。问有多少个区间是被包含了。

水。。。

按x排升序,然后对于每个区间,看前面已经查询过的区间的y的最大值,这个用一个变量保存就可以了。

鄙视,数组开小了,看成了10000,另外,貌似sort直接对pair排序是默认以first作为第一关键字的。

pair<int,int>p[maxn];

////// without this ,the code also get a Accepted,and the runtime is 90ms;   why;
/*bool cmp(pair<int,int>x,pair<int,int>y){
return x.first< y.first;
}*/

int main()
{
int n,i;
while(cin>>n){
for(i=1;i<=n;++i){
scanf("%d%d",&p[i].first,&p[i].second);
}
sort( p+1, p+1+n );
int ans=0,maxval= p[1].second;
for(i=2;i<=n;++i){
if( p[i].second < maxval )
ans++;
else
maxval= p[i].second;
}
cout<< ans <<endl;
}
}


D题

给定长度为n的字符串,n<=500,然后是k,要求把串划分为 不多于k个段,可以是1,2,3..k段,要求每段是回文串。 同时为了保证是回文串,需要进行必要的修改字符操作,要求输出最小的修改次数,即最后的结果串,段段之间用+号连接,如果多个答案,随意输出一个。

如:

abdcaba
5

answer:

0
a+b+d+c+aba


我的做法是dp,为了方便,我的字符串从1开始到n作为索引,dp[i][j]: [1,i] 划分为 j段的最小修改次数,显然dp[i][i] = 0;

dp[i][j]= min( dp[k][j-1]+ cal( i, i-k ));

cal( i, i-k ): 以i结尾长度为i-k的子串成为回文串的最小修改次数。

cal()可以预处理出来,cal(), 和DP都是 O( 500*500*500 )的复杂度,应该可以用dp优化;

同时为了记录+号的位置,pos[i][j]: 表示dp[i][j]取得最优值时,i前面第一个+出现在哪个字符后后面。

E题

给定长度n<= 200000的串,要求最长的满足v<= 2*c 的子串长度及相应的个数。v表示元音字母个数,c表示辅音字母个数;

我的考虑是 如果v[i] <= 2*c[i], 那么用i更新ans,这是肯定的。

如果不满足上式呢? 有 v[i] - v[j] <= 2* ( c[i]- c[j] ): ==> 2*c[j]- v[j] <= 2*c[i] - v[i];

也即我们需要找到比i前面 比 (2*c[i]-v[i]) 小或相等的 j,且取最小的那个,解决方法是以 2*c[i]-v[i] 作为索引,以i作为值建立线段树维护最小值;

又考虑到 v[i] <= 2*c[i], 那么用i更新ans , 建立线段树是针对 v[i] > 2*c[i]的i,且值是负数,可以加上一个add= n+1;,这样总区间是

[ 1,n ],。

当然会有更方便写的数据结构,我一时没有想到。

具体见代码。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: