字符串消除-庞果网题目

本文由 @lonelyrains 出品,转载请注明出处。 
文章链接: http://blog.csdn.net/lonelyrains/article/details/12969177


题目链接: http://hero.pongo.cn/Question/Details?ID=85&ExamID=83

解决方法:贪心法

详细描述:每次从字符串选择两个相邻可消除字符,要求是这两个字符都是当前最多和次多的。例如abbcabca,a和b各有三个,而c只有两个。就先消除ab,转成c。只转一个字符,然后重新统计整个字符串cbcabca,c有三个,而a和b都是两个,但是先搜到cb,所以转cb为a。

原理证明:要求最终生成的字符串最短,肯定是被消除的次数最多,而且最终生成的字符串肯定只有一种字符。要达到最终消除次数最多,每次消除当前最多的字符,最好同时消除次多的字符,避免出现多个相同字符连续的情况,即可。假设不消除最多的,那么可能出现生成生成更多的最多字符,从而最终不能消除。

具体实现(已提交通过):

#include <cstdio>
#include <cstdio>
#include <string>
#include <cstring>
#include <ctime>
#include <cstdlib>

using namespace std;

int max(int a,int b,int c)
{
    int max = 0;
    if(a>b)
    {
        max = a;
    }
    else
    {
        max = b;
    }
    if(max<c)
    {
        max = c;
    }
    return max;
}

char clearCh(char a,char b)
{
    if(a==b)
    {
        return a;
    }
    if((a == 'a' && b == 'b') || (a == 'b' && b == 'a'))
    {
        return 'c';
    }
    if((a == 'b' && b == 'c') || (a == 'c' && b == 'b'))
    {
        return 'a';
    }
    if((a == 'a' && b == 'c') || (a == 'c' && b == 'a'))
    {
        return 'b';
    }
    return a;
}

void clearFirst(const char *s,char *p,char a,char b)
{
    int i=0;
    int clearFlag = 0;
    for(i=0;i<strlen(s)-1;i++)
    {
        if((s[i]==a && s[i+1]==b)||(s[i]==b && s[i+1]==a))
        {
            strncpy(p,s,i);
            *(p+i) = clearCh(a,b);
            strncpy(p+i+1,s+i+2,strlen(s)-i-2);
            clearFlag = 1;
            break;
        }
    }
    if(clearFlag == 0)
    {
        for(i=0;i<strlen(s)-1;i++)
        {
            if((s[i]==a && s[i+1]!=a)||(s[i]!=a && s[i+1]==a))
            {
                strncpy(p,s,i);
                *(p+i) = clearCh(s[i],s[i+1]);
                strncpy(p+i+1,s+i+2,strlen(s)-i-2);
                clearFlag = 1;
                break;
            }
        }
    }
    //printf("%s\n",p);
}

int minLength(const char *s) 
{
    int a=0,b=0,c=0,i=0,minLen=0;
    for(i=0;i<strlen(s);i++)
    {
        if(s[i] == 'a')
        {
            a++;
        }
        if(s[i] == 'b')
        {
            b++;
        }
        if(s[i] == 'c')
        {
            c++;
        }
    }
    
    if(a == 0 && b == 0)
    {
        minLen =  c;
    }
    else if(b == 0 && c == 0)
    {
        minLen =  a;
    }
    else if(a == 0 && c == 0)
    {
        minLen =  b;
    }
    else
    {
        char *p = new char[strlen(s)];
        if(p==NULL)
        {
            return -1;
        }
        memset(p,0,sizeof(char)*(strlen(s)));
        if(a>=b && b>=c)
        {
            clearFirst(s,p,'a','b');
        }
        else if(a>=c && c>=b)
        {
            clearFirst(s,p,'a','c');
        }
        else if(b>=a && a>=c)
        {
            clearFirst(s,p,'b','a');
        }
        else if(b>=c && c>=a)
        {
            clearFirst(s,p,'b','c');
        }
        else if(c>=b && b>=a)
        {
            clearFirst(s,p,'c','b');
        }
        else if(c>=a && a>=b)
        {
            clearFirst(s,p,'c','a');
        }
        minLen = minLength(p);
        delete(p);
    }

    return minLen;
}



//start 提示:自动阅卷起始唯一标识,请勿删除或增加。
int main()
{   
    return 0;
} 
//end //提示:自动阅卷结束唯一标识,请勿删除或增加。


评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值