单调递增的数字

力扣题目链接(opens new window)

给定一个非负整数 N,找出小于或等于 N 的最大的整数,同时这个整数需要满足其各个位数上的数字是单调递增。

(当且仅当每个相邻位数上的数字 x 和 y 满足 x <= y 时,我们称这个整数是单调递增的。)

示例 1:

  • 输入: N = 10
  • 输出: 9

示例 2:

  • 输入: N = 1234
  • 输出: 1234

示例 3:

  • 输入: N = 332
  • 输出: 299

说明: N 是在 [0, 10^9] 范围内的一个整数

思路

题意解析

题目举的例子有点不太好理解,这里再举一个吧

  • 输入: N = 32
  • 输出: 29

什么意思呢?就是32这个输入数,其不满足各个位数上的数字是单调递增的规则

因此需要找一个比32小但是尽可能大的整数,该整数还要满足各个位数上的数字单调递增的规则

那么29显然是符合条件的

解法

用人脑想可以想到上面情况中29是符合条件的数,但是用算法怎么实现?

如果去遍历nums(32),可以得到nums[i](2)小于nums[i - 1](3)

此时出现了单调递减,不符合条件,需要去找比nums小的一个最大单调递增整数

怎么找?

因为32已经是单调递减了,所以个位上的2无论怎么变化都不能逆转递减的趋势,因此需要把十位减小使整体满足递减的要求,按照上述逻辑可以得到22

因为大于等于都算递增,所以此时的22已经满足单调递增的条件了,接下来需要达成最大整数这个条件

显然22不是小于32的数中满足单调递增的最大整数,在2X中找最大整数,那X只要取9就可以使整体最大,所以满足所有条件的数是29

总结一下:

​ 当遍历遇到单调递减的两位数时,可以直接将十位数减1,然后个位取9,得到小于当前这两位数的最大单调递增整数

​ 这里也可以确认 ,遍历时我们是两个两个数来判断是否符合条件的

遍历nums的顺序

问题又来了,遍历nums的顺序是什么呢?

先说结论,应该从后先前遍历

举个例子:

  • 输入: N = 332
  • 输出: 299

如果从前向后遍历,过程如下:

​ 先得到33,33满足单调递增条件,不管

​ 然后得到32,不满足条件,按照上面的逻辑找到小于32有满足条件的最大整数29,遍历结束

​ 此时整体为329,还是不满足条件

​ 原因是从前向后遍历没有将前面判断得到的结果利用起来

在来看从后向前遍历:

​ 先得的32,不满足条件,修改为29,此时整体为329

​ 然后得到32(基于前一步的结果),还是不满足条件,修改为29,此时整体为299,遍历结束

​ 结果正确

修改9的逻辑

上面其实就是全部的解题逻辑了,但是按照上面的讨论去实现代码时会发现有问题,每次遇到单调递减的数时我们会下意识的直接把9给换到相应位置,但其实这样会出错

举个例子:

  • 输入: N = 3232
  • 输出: 2999

这个例子的正确答案是2999,如果在实现时按照上面的问题逻辑来的话,得到的结果会是2929

先不说它是不是最大的吧,它就不满足单调递增

因此我们需要一个变量,记录最后一次要改9的位置,然后将其后位置的所有数全部改成9

3  2  3  2
↑ ↑
i-1 i

i 处要改9,此时,flag == 3

3  2  3  2
↑ ↑
i-1 i

此时flag还是3

 3  2  3  2
↑ ↑
i-1 i

i 处要改9,此时,flag == 1

遍历结束,将flag往后的所有数都改成9,得到正确结果2999

注意,flag的初始值应该指向数组末尾

代码

注意这里给的输入是一个int,要先把其转化为str才能遍历

然后改完记得再转回来(使用stoi函数

class Solution {
public:
int monotoneIncreasingDigits(int n) {
string strNum = to_string(n);
int flag = strNum.size();//flag初始值应该指向数组末尾
for(int i = strNum.size() - 1; i > 0; --i){
if(strNum[i - 1] > strNum[i]){//单调递减
strNum[i - 1]--;
// strNum[i] = '9';
flag = i;//记录要修改9的位置
}
}//遍历完成,找到要修改9的位置 for(int i = flag; i < strNum.size(); ++i){//将flag后的数全改为9
strNum[i] = '9';
} int num = stoi(strNum);//转回int
return num;
}
};

【LeetCode贪心#11】单调递增的数字(详解)的更多相关文章

  1. 【LeetCode】738. 单调递增的数字

    738. 单调递增的数字 知识点:字符串:贪心 题目描述 给定一个非负整数 N,找出小于或等于 N 的最大的整数,同时这个整数需要满足其各个位数上的数字是单调递增. (当且仅当每个相邻位数上的数字 x ...

  2. Java实现 LeetCode 738 单调递增的数字(暴力)

    738. 单调递增的数字 给定一个非负整数 N,找出小于或等于 N 的最大的整数,同时这个整数需要满足其各个位数上的数字是单调递增. (当且仅当每个相邻位数上的数字 x 和 y 满足 x <= ...

  3. C++11 并发指南四(<future> 详解二 std::packaged_task 介绍)

    上一讲<C++11 并发指南四(<future> 详解一 std::promise 介绍)>主要介绍了 <future> 头文件中的 std::promise 类, ...

  4. C++11 并发指南四(<future> 详解三 std::future & std::shared_future)

    上一讲<C++11 并发指南四(<future> 详解二 std::packaged_task 介绍)>主要介绍了 <future> 头文件中的 std::pack ...

  5. C++11 并发指南三(Lock 详解)(转载)

    multithreading 多线程 C++11 C++11多线程基本使用 C++11 并发指南三(Lock 详解) 在 <C++11 并发指南三(std::mutex 详解)>一文中我们 ...

  6. C++11 并发指南四(<future> 详解三 std::future & std::shared_future)(转)

    上一讲<C++11 并发指南四(<future> 详解二 std::packaged_task 介绍)>主要介绍了 <future> 头文件中的 std::pack ...

  7. Spring框架系列(11) - Spring AOP实现原理详解之Cglib代理实现

    我们在前文中已经介绍了SpringAOP的切面实现和创建动态代理的过程,那么动态代理是如何工作的呢?本文主要介绍Cglib动态代理的案例和SpringAOP实现的原理.@pdai Spring框架系列 ...

  8. [Swift]LeetCode738. 单调递增的数字 | Monotone Increasing Digits

    Given a non-negative integer N, find the largest number that is less than or equal to Nwith monotone ...

  9. C++11 并发指南三(Lock 详解)

    在 <C++11 并发指南三(std::mutex 详解)>一文中我们主要介绍了 C++11 标准中的互斥量(Mutex),并简单介绍了一下两种锁类型.本节将详细介绍一下 C++11 标准 ...

  10. C++11 并发指南------std::thread 详解

    参考: https://github.com/forhappy/Cplusplus-Concurrency-In-Practice/blob/master/zh/chapter3-Thread/Int ...

随机推荐

  1. Miller-Rabin素性判定算法

    Miller-Rabin素性判定算法是一种基于概率的判定算法,每次判定n是素数的正确性概率至少为75%,出错的概率小于25%. 如果对n进行k次素性检测,如果结果n为素数,那么n为合数的概率为1/(4 ...

  2. 再见IE

  3. nchu第二次面向对象编程博客作业

    前言:   本次博客包含的内容有pta题目集4(四边形).5(五边形)以及期中考试三次题目集.其中第四次和第五次题目集难度较大,比较复杂,涉及的知识点也比较多.而期中考试由于是在课堂上完成,难度较小, ...

  4. formidable处理文件上传的细节

    koa在请求体的处理方面依赖于通用插件koa-bodyparser或者koa-body,前者比较小巧,内部使用了co-body库,可以处理一般的x-www-form-urlencoded格式的请求,但 ...

  5. C# Async / Await State Machine

    The async/await keywords in C# are very much syntactical sugar that the compiler will use to generat ...

  6. 通过Linux的socket套接字实现客户端与服务器端的通信

    具体案例:使用树莓派ds18b20温度传感器实现温度上报 首先需要获得传感器文件中保存的温度信息: 温度信息通常保存在路径为"/sys/bus/w1/devices/28-xxxxxxxxx ...

  7. 线程池使用、countDownLatch、以及数据库批量插入 添加配置优化插入与计算

    //新建线程池ThreadPoolExecutor cpuThreadPoolExecutor = ThreadUtil.getCpuThreadPoolExecutor(); //使用Countdo ...

  8. pytorch基础 自动求导

    1.把pytorch当成是numpy来用就行 2. 一个典型的张量是这样定义的. import pytorch as tt n=tt.tensor([1,2,3],dtype=True,requirg ...

  9. ajax json php post 方法

    1.前端 <script type="text/javascript"> function LoadData(arg){ arg.dept=$("#DeptS ...

  10. 基于airtest验证Android端app是否安装及自动化安装

    ​ 1.检测app是否安装: 使用check_app方法检测是否安装:为什么需要在封装一层做断言呢?主要check_app方法安装成功会返回True,但是未检测到安装时直接报异常了,停止执行.无法直接 ...