[暴力枚举] CF118C Fancy Number

题目传送门

思路

暴力枚举

这个题目的核心思想其实就是暴力枚举,并且这个题细节特别多,要特别注意。我整整调了1个小时

就是枚举一边 0 0 0 9 9 9,并求出使得车牌号码中有 k k k 个枚举的数的代价是多少,如果这个代价小于以前枚举出来的最小代价,则这个代价就成为最小代价,并将方案记录下来。 如过以前的最小代价等于当前的代价,则就要注意一下字典序。

详情见代码。

代码

#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
int n,k;
int a[15],q[100005],ans=0x3f3f3f3f,now[100005],now2[100005];
int flg;//flg来标记最后的答案是由1~9里面的那一个数字产生的 
int cnt[100005];//标记要改变成flg的数
struct node
{
    int id,num,w,pqr;
    bool operator <(const node &n)const//特别注意一下这里的排序规则(注意字典序)。
    {
        if(pqr<=w&&pqr<=n.w)
            return num<n.num||(num==n.num&&id<n.id);
        else if(pqr>=w&&pqr>=n.w)
            return num<n.num||(num==n.num&&id>n.id);
        else
            return num<n.num||(num==n.num&&w>n.w);
    }
}arr[100005];
char p[100005];
int main()
{
	//输入
    cin>>n>>k;
    for(int i=1;i<=n;++i)
    {
        cin>>p[i];
        q[i]=p[i]-'0';
        a[q[i]]++;//桶来标记1~9之间共有多少个数字
    }
    for(int i=0;i<=9;++i)
    {
        if(a[i]>=k)//如果不用改变数字就可以完成了
        {
            cout<<0<<endl;
            for(int i=1;i<=n;++i)
                cout<<q[i];
            return 0;
        }
        for(int j=1;j<=n;++j)
        {
            arr[j].id=j;//id表示是这个车牌号的第几个
            arr[j].w=q[j];//表示车牌号上的数值
            arr[j].num=abs(q[j]-i);//表示替换这个数字为i需要多少代价
            arr[j].pqr=i;//记录最后要改变成什么
        }
        sort(arr+1,arr+n+1);//通过排序规则进行排序
        int bj=0;//来改变为车牌号有k个i的最小代价
        for(int j=1;j<=k;++j)
        {
            bj+=arr[j].num;
        }
        if(ans>bj)//如果以前的最小代价比当前最小代价大
        {
            ans=bj;//改变最小代价
            flg=i;//最后的答案是由车牌号变成k个i产生的
            memset(cnt,0,sizeof(cnt));
            for(int j=1;j<=k;++j)
            {
                cnt[j]=arr[j].id;//记录究竟要改变那些下标上的数字为i
            }
        }
        else if(ans==bj)//如果以前的最小代价等于当前最小待机,则我们要注意字典序
        {
            bool kkk=0;//标记
            for(int j=1;j<=n;++j)
            {
            	//用一个临时调用的数字来进行操作
                now[j]=q[j];
                now2[j]=q[j];
            }
            for(int j=1;j<=k;++j)
            {
            	//通过储存下来的要改变的方案进行改变
                now[arr[j].id]=i;
                now2[cnt[j]]=flg;
            }
            for(int j=1;j<=n;++j)
            {
                if(now[j]<now2[j])//如果用i产生的最小代价的字典序更小
                {
                    kkk=1;//标记
                    break;//跳出循环
                }
                else if(now[j]>now2[j])//如果用flg产生的最小代价的字典序更小
                {
                    break;//跳出循环,无需标记
                }
            }
            if(kkk==1)//如果用i产生的最小代价的字典序更小
            {
                flg=i;//flg进行替换
                memset(cnt,0,sizeof(cnt));
                for(int j=1;j<=k;++j)
                {
                    cnt[j]=arr[j].id;//储存那些下标需要替换
                }
            }
        }
    }
    for(int i=1;i<=k;++i)
        q[cnt[i]]=flg;//进行改变
    //输出
    cout<<ans<<endl;
    for(int i=1;i<=n;++i)
        cout<<q[i];
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值