题目描述

求出1~13的整数中1出现的次数,并算出100~1300的整数中1出现的次数?为此他特别数了一下1~13中包含1的数字有1、10、11、12、13因此共出现6次,但是对于后面问题他就没辙了。ACMer希望你们帮帮他,并把问题更加普遍化,可以很快的求出任意非负整数区间中1出现的次数(从1 到 n 中1出现的次数)。

思路

像类似这样的问题,我们可以通过归纳总结来获取相关的东西。

首先可以先分类:

个位
我们知道在个位数上,1会每隔10出现一次,例如1、11、21等等,我们发现以10为一个阶梯的话,每一个完整的阶梯里面都有一个1,例如数字22,按照10为间隔来分三个阶梯,在完整阶梯0-9,10-19之中都有一个1,但是19之后有一个不完整的阶梯,我们需要去判断这个阶梯中会不会出现1,易推断知,如果最后这个露出来的部分小于1,则不可能出现1(这个归纳换做其它数字也成立)。

我们可以归纳个位上1出现的个数为:

n/10 * 1+(n%10!=0 ? 1 : 0)

十位
现在说十位数,十位数上出现1的情况应该是10-19,依然沿用分析个位数时候的阶梯理论,我们知道10-19这组数,每隔100出现一次,这次我们的阶梯是100,例如数字317,分析有阶梯0-99,100-199,200-299三段完整阶梯,每一段阶梯里面都会出现10次1(从10-19),最后分析露出来的那段不完整的阶梯。我们考虑如果露出来的数大于19,那么直接算10个1就行了,因为10-19肯定会出现;如果小于10,那么肯定不会出现十位数的1;如果在10-19之间的,我们计算结果应该是k - 10 + 1。例如我们分析300-317,17个数字,1出现的个数应该是17-10+1=8个。

那么现在可以归纳:十位上1出现的个数为:

设k = n % 100,即为不完整阶梯段的数字
归纳式为:(n / 100) * 10 + (if(k > 19) 10 else if(k < 10) 0 else k - 10 + 1)
百位
现在说百位1,我们知道在百位,100-199都会出现百位1,一共出现100次,阶梯间隔为1000,100-199这组数,每隔1000就会出现一次。这次假设我们的数为2139。跟上述思想一致,先算阶梯数 * 完整阶梯中1在百位出现的个数,即n/1000 * 100得到前两个阶梯中1的个数,那么再算漏出来的部分139,沿用上述思想,不完整阶梯数k199,得到100个百位1,100<=k<=199则得到k - 100 + 1个百位1。

那么继续归纳百位上出现1的个数:

设k = n % 1000
归纳式为:(n / 1000) * 100 + (if(k >199) 100 else if(k < 100) 0 else k - 100 + 1)
后面的依次类推....

再次回顾个位
我们把个位数上算1的个数的式子也纳入归纳式中

k = n % 10
个位数上1的个数为:n / 10 * 1 + (if(k > 1) 1 else if(k < 1) 0 else k - 1 + 1)
完美!归纳式看起来已经很规整了。 来一个更抽象的归纳,设i为计算1所在的位数,i=1表示计算个位数的1的个数,10表示计算十位数的1的个数等等。

k = n % (i * 10)
count(i) = (n / (i * 10)) * i + (if(k > i * 2 - 1) i else if(k < i) 0 else k - i + 1)
好了,这样从10到10的n次方的归纳就完成了。

牛客网链接

js代码

function NumberOf1Between1AndN_Solution(n)
{
// write code here
if (n <= 0) return 0
let mod
let res = 0
for (let i = 1; i <= n; i *= 10) {
res += Math.floor(n / (i*10)) * i
mod = n % (i*10)
if (mod > i * 2 -1) res += i
else if (mod >= i && mod <= i * 2 -1 ) res += (mod - i + 1)
}
return res
}```

最新文章

  1. ASP.NET页面传值方式
  2. Linux系统监控实用工具Glances
  3. iOS视频压缩处理
  4. MyBatis双数据源配置
  5. Spring集成Redis缓存
  6. idea编译错误提示编译版本不对,需要注意的配置
  7. springMVC使用HandlerMethodArgumentResolver 自定义解析器实现请求参数绑定方法参数
  8. strstr函数的运用
  9. java 线程(四)线程安全 同步方法
  10. java1.8新特性(optional 使用)
  11. Python OS 文件/目录方法
  12. 关于前一篇innodb自增列自己的一点补充
  13. Windows常用内容渗透命令
  14. IOS 上传下载
  15. mvc上传图片(上传和预览)webuploader
  16. Elastic Search操作入门
  17. C# 注册Dll文件
  18. 牛客练习赛16 F 选值【二分/计数】
  19. linux应用之yum命令的软件源的更换(centos)
  20. liunx 安装工具总结

热门文章

  1. MongoDB学习笔记二:使用Docker安装MongoDB
  2. 【err】tensorflow.python.framework.errors_impl.OutOfRangeError: RandomShuffleQueue
  3. [LeetCode] 61. Rotate List 旋转链表
  4. [LeetCode] 114. Flatten Binary Tree to Linked List 将二叉树展平为链表
  5. [LeetCode] 268. Missing Number 缺失的数字
  6. 【视频开发】【计算机视觉】doppia编译之一:前言及安装CUDA
  7. nuxt/eapress 安装报错Module build failed: ValidationError: PostCSS Loader Invalid OptionsModule build failed: ValidationError: PostCSS Loader Invalid Options options[&#39;useConfigFile&#39;] is an invalid additi
  8. 2、Maven的简介和配置
  9. TCP/IP学习笔记16--TCP--特点,数据重发,连接管理,段
  10. 【LeetCode】两数之和【优化查询过程即可】