[HEOI2015]定价 (贪心)-程序员宅基地

分类讨论大法好!

\(solution:\)

先说一下我对这个题目的态度:

首先这一题是贪心,这个十分明显,看了一眼其他题解都是十分优秀的贪心,可是大家都没有想过吗:你们贪心都是在区间\([l,r]\)上枚举的贪心,虽然每一次可以直接加上 10 的阶乘,但你们毕竟是用的\(int\)\(long long\) 啊!。这一题得正解复杂度是\(log_{10}(n)\)的,不得不说题目的数据范围给的太小了只有\(10^9\) ,如果按照复杂度,数据范围甚至可以到\(10^{10000000}\)级别,这个时候你们难道还能在区间\([l,r]\)上枚举吗?(高精怕都要超时!)

所以虽然分类讨论十分麻烦,甚至代码很长,但它才应该是真正意义上的正解啊!(当然,考场见机行事)

好的,不做“推销”了,我们进入正文。

首先这一题,我们可以用字符串的方式读入两个端点值。然后如果我们仔细读题,就可以发现一些贪心方案

  1. 我们要让最终的价格数的高位上不是0的数尽可能的少!(当然,要在末尾全是0的基础上)
  2. 只有当不是0的数确定下来,我们再来判断最后一个不是0的数能否为5!
  3. 如果最后一个不是0的数不能为5,我们就要让最后一个不是0的数尽可能的小!
  4. 让最终得到的价格数尽可能的小!

开始讨论之前先让你们有个思维准备:先看看下面代码有多少个\(continue\),我们大概就要讲多少种情况。首先我们准备一个快写函数:将字符串\(s\)\(j-1\)位输出,然后将第\(j\)为换成字符\(k\),再输出\(n\)个0

然后为了涵盖多种情况,我们发现如果两个数的长度不一样,是很难讨论的。于是我们分三种情况:(两个数长度差了2位或以上)(两个数的长度差为1)(两个数长度相同)

两个数长度差了2位或以上:(这个想一下,我们发现只有两种情况

  1. 如果较小的数的最高位比5小,就换成5输出(要特判5,50,500,5000......)
  2. 否则将较小的数的最高位换成5,再乘上10输出。

两个数的长度差为1:(比上面要多一种情况)

  1. 如果较小的数的最高位比5小,就换成5输出(要特判5,50,500,5000......)
  2. 如果较大的数的最高位大于等于5,将较大的数的最高位换成5,输出。
  3. 将较小的的数的最高位上的数加1,然后全接0.(要特判如10,60,300等(高位不用加1))

两个数长度相同:这个我们仔细想一下就会发现如果这两个数的高位相同,那么这些相同的高位可以直接输出了,所以我们放个循环\(O(n)\)找到不同的那一位,然后我们发现这一位后面的数可以忽略了(一定会被填成0的,因为贪心第一点:要让最终的价格数的高位上不是0的数尽可能的少)而针对这一位能否填成5,我们又有四中情况:

  1. 这两个数完全相同,这个直接输出
  2. 较小的数的这一位小于5,较大的数的这一位大于5,将这一位改成5输出(不需要特判5,50...这个归到下一类)
  3. 较小的数这一位后面全是0,直接输出
  4. 较小的数这一位后面不全是0,将这一位+1,然后输出。
平心而论:这一题情况并不算多(详见猪国杀)

对了,我们上文有提到一些需要特判的东西,这个只需要,在开始时用一个变量\(f\)来记录较小的数从那一位开始之后全是0即可!

\(code:\)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define rg register int

using namespace std;

int t,a,b,f;
char l[11];
char r[11];

inline void qw(char s[],int j,char k,int n){//有这个函数输出就方便多了
    for(rg i=0;i<j;++i)putchar(s[i]);
    k>'9'?printf("10"):putchar(k);
    for(rg i=1;i<=n;++i)putchar('0');
    puts("");
}

int main(){ cin>>t;
    //freopen("Price.in","r",stdin);
    //freopen("Price.out","w",stdout);
    while(t--){
        cin>>l>>r;
        f=a=strlen(l)-1;
        b=strlen(r)-1;
        while(l[f]=='0')--f;//记录从那一位开始后面全是0
        if(b-a>1){//差一位以上
            if(l[0]<'5'||(l[0]=='5'&&f==0)){qw(l,0,'5',a);continue;}
            qw(l,0,'5',a+1);continue;
        }if(b-a>0){//差一位
            if(l[0]<'5'||(l[0]=='5'&&f==0)){qw(l,0,'5',a);continue;}
            if(r[0]>='5'){qw(l,0,'5',b);continue;}
            if(f==0){qw(l,0,l[0],a);continue;}
            qw(l,0,l[0]+1,a);continue;
        }int i=0;while(l[i]==r[i]&&i<=a)++i;//找到不同的哪一位
        if(i>a||f<i){qw(l,a,l[a],0);continue;}
        if(l[i]<'5'&&r[i]>='5'){qw(l,i,'5',a-i);continue;}
        if(f==i){qw(l,i,l[i],a-i);continue;}
        qw(l,i,l[i]+1,a-i);
    }
    return 0;
}

情况都是按上文顺序写的,但本人还是很菜可能思虑不周,欢迎大家来hack(hack成功了麻烦提醒一下,O(∩_∩)O谢谢)

转载于:https://www.cnblogs.com/812-xiao-wen/p/10333583.html

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_30872157/article/details/96780410

智能推荐

c# 调用c++ lib静态库_c#调用lib-程序员宅基地

文章浏览阅读2w次,点赞7次,收藏51次。四个步骤1.创建C++ Win32项目动态库dll 2.在Win32项目动态库中添加 外部依赖项 lib头文件和lib库3.导出C接口4.c#调用c++动态库开始你的表演...①创建一个空白的解决方案,在解决方案中添加 Visual C++ , Win32 项目空白解决方案的创建:添加Visual C++ , Win32 项目这......_c#调用lib

deepin/ubuntu安装苹方字体-程序员宅基地

文章浏览阅读4.6k次。苹方字体是苹果系统上的黑体,挺好看的。注重颜值的网站都会使用,例如知乎:font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Microsoft YaHei, Source Han Sans SC, Noto Sans CJK SC, W..._ubuntu pingfang

html表单常见操作汇总_html表单的处理程序有那些-程序员宅基地

文章浏览阅读159次。表单表单概述表单标签表单域按钮控件demo表单标签表单标签基本语法结构<form action="处理数据程序的url地址“ method=”get|post“ name="表单名称”></form><!--action,当提交表单时,向何处发送表单中的数据,地址可以是相对地址也可以是绝对地址--><!--method将表单中的数据传送给服务器处理,get方式直接显示在url地址中,数据可以被缓存,且长度有限制;而post方式数据隐藏传输,_html表单的处理程序有那些

PHP设置谷歌验证器(Google Authenticator)实现操作二步验证_php otp 验证器-程序员宅基地

文章浏览阅读1.2k次。使用说明:开启Google的登陆二步验证(即Google Authenticator服务)后用户登陆时需要输入额外由手机客户端生成的一次性密码。实现Google Authenticator功能需要服务器端和客户端的支持。服务器端负责密钥的生成、验证一次性密码是否正确。客户端记录密钥后生成一次性密码。下载谷歌验证类库文件放到项目合适位置(我这边放在项目Vender下面)https://github.com/PHPGangsta/GoogleAuthenticatorPHP代码示例://引入谷_php otp 验证器

【Python】matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距-程序员宅基地

文章浏览阅读4.3k次,点赞5次,收藏11次。matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距

docker — 容器存储_docker 保存容器-程序员宅基地

文章浏览阅读2.2k次。①Storage driver 处理各镜像层及容器层的处理细节,实现了多层数据的堆叠,为用户 提供了多层数据合并后的统一视图②所有 Storage driver 都使用可堆叠图像层和写时复制(CoW)策略③docker info 命令可查看当系统上的 storage driver主要用于测试目的,不建议用于生成环境。_docker 保存容器

随便推点

网络拓扑结构_网络拓扑csdn-程序员宅基地

文章浏览阅读834次,点赞27次,收藏13次。网络拓扑结构是指计算机网络中各组件(如计算机、服务器、打印机、路由器、交换机等设备)及其连接线路在物理布局或逻辑构型上的排列形式。这种布局不仅描述了设备间的实际物理连接方式,也决定了数据在网络中流动的路径和方式。不同的网络拓扑结构影响着网络的性能、可靠性、可扩展性及管理维护的难易程度。_网络拓扑csdn

JS重写Date函数,兼容IOS系统_date.prototype 将所有 ios-程序员宅基地

文章浏览阅读1.8k次,点赞5次,收藏8次。IOS系统Date的坑要创建一个指定时间的new Date对象时,通常的做法是:new Date("2020-09-21 11:11:00")这行代码在 PC 端和安卓端都是正常的,而在 iOS 端则会提示 Invalid Date 无效日期。在IOS年月日中间的横岗许换成斜杠,也就是new Date("2020/09/21 11:11:00")通常为了兼容IOS的这个坑,需要做一些额外的特殊处理,笔者在开发的时候经常会忘了兼容IOS系统。所以就想试着重写Date函数,一劳永逸,避免每次ne_date.prototype 将所有 ios

如何将EXCEL表导入plsql数据库中-程序员宅基地

文章浏览阅读5.3k次。方法一:用PLSQL Developer工具。 1 在PLSQL Developer的sql window里输入select * from test for update; 2 按F8执行 3 打开锁, 再按一下加号. 鼠标点到第一列的列头,使全列成选中状态,然后粘贴,最后commit提交即可。(前提..._excel导入pl/sql

Git常用命令速查手册-程序员宅基地

文章浏览阅读83次。Git常用命令速查手册1、初始化仓库git init2、将文件添加到仓库git add 文件名 # 将工作区的某个文件添加到暂存区 git add -u # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件git add -A # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件...

分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120-程序员宅基地

文章浏览阅读202次。分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120

【C++缺省函数】 空类默认产生的6个类成员函数_空类默认产生哪些类成员函数-程序员宅基地

文章浏览阅读1.8k次。版权声明:转载请注明出处 http://blog.csdn.net/irean_lau。目录(?)[+]1、缺省构造函数。2、缺省拷贝构造函数。3、 缺省析构函数。4、缺省赋值运算符。5、缺省取址运算符。6、 缺省取址运算符 const。[cpp] view plain copy_空类默认产生哪些类成员函数

推荐文章

热门文章

相关标签