题目

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

思考

方法1:

依次从 1 到 n 遍历,统计每个数中 1 出现的次数,然后累加起来。时间复杂度:遍历 n 个数,每次统计可以在 log(n) 时间内完成,因此 O(nlog(n))。

方法2:

新的角度:给定 N,统计“小于 N 的数在每一位上可能出现 1 的次数” 之和,可得到最终的解。使用函数 f(N) 表示。

1 位数的情况

如果 N=4,则所有可考虑的数为: 1,2,3,4 由此可得 f(N) = 1。不难看出对于所有 1<=N<=9, f(N) = 1,特殊的有 f(0)=0。

2 位数的情况

假设 N=14,可考虑的数为:1,2,3,4,5,6,7,8,9,10,11,12,13,14。

个位上出现 1 的情况:1,11; 2个

十位上出现 1 的情况:10, 11, 12, 13, 14;5个

综合:f(N)=个位数的情况 + 十位数的情况 = 2 + 5 = 7

一般的,

  • 个位上 1 的个数统计可以有两种情况:

    • 情况 1 : 当前数字 N 个位数是 0,则个位上 1 出现的次数为十位出现的数。(比如 N=10,个位上出现 1 的次数是 1)
    • 情况 2 :当前数字 N 个位数大于 0, 则个位上 1 出现的次数为十位出现的数加 1。(比如 N=14,个位上出现 1 的次数是 1+1=2,及 1 和 11)
  • 十位上 1 的个数统计:
    • 情况 1 : 当十位上的数为 1 时,则十位上出现 1 的次数为:个位上的数加 1 (比如 N=14, 十位上出现 1 的次数 = 4+1=5,及:10,11,12,13,14)
    • 情况 2 :当十位上的数大于 1 时,则十位上出现 1 的次数为 10 次。(比如 N=24, 十位上出现 1 的次数 = 10,及:10,11,12,13,14,15,16,17,18,19)

其他两位数字可以类似的方法很快计算出来

3 位数的情况

假设 N = 132

  • 个位上出现 1 的次数为 14 次:1,11,21,...,91,101,111,121,131
  • 十位上出现 1 的次数为 20 次:10~19, 110~119
  • 百位上出现 1 的次数为 33 次:100~132

一般化

同理可以计算 4,5,甚至 k 位数中 1 的个数。

一般的,假设有数 N = ak... a1

我们可以将数字分成 3 个部分,高位部分,当前计算的位置,低位部分;如果计算第 i 位上出现 1 的次数,则高位部分为 ak...ai+1,当前计算位置 ai,低位部分 ai-1...a1。而第 i 位上出现 1 的次数所受影响的来源也将分配到这 3 个部分当中。

在来一个例子,假设 N = 13014

  • 个位出现 1 的次数,ihight=1301, icur = 4,因此 icur > 1,所以 1 的次数为 ihight+1 = 1302
  • 十位出现 1 的次数,ihight=130, icur=1, ilow=4,因此 1 的次数为 ihight*10+ilow+1=130*10+4+1=1305
  • 百位出现 1 的次数,ihight=13, icur=0, ilow=14
    • icur = 0,影响只来源于高位部分,13 个 100,每个100在百位上将会出现100个1,也就是 13*100=1300
    • icur = 1,假设 N=13114,影响来源于高位和低位:
      • 高位部分,ihight=13, 13 个 100,则 1 的个数为 13 * 100 = 1300
      • 低位部分,ilow=14,13100~13114,1 的个数为 ilow+1 = 14+1 = 15
    • icur > 1,假设 N=13214,影响仅来源于高位:
      • ihight = 13,13 个 100,当前尾部 13100~13199 有一个100,总攻是 14 个,及 ihight+1 个 100,1 的个数为 (13+1)*100 = 1400

由此规律很容易实现代码。

思考总线,从简单的例子入手找规律,抽象出一般化的规律,要注意细节

code


#include <iostream>
#include <vector>
#include <algorithm>

using namespace std;

class Solution{
public:
    int NumberOf1Between1AndN_Solution(int n){
        int icout = 0;
        int ihight = 0;
        int ilow = 0;
        int icur = 0;
        int ifactor = 1;

        while( n / ifactor != 0){
            ihight = n / (ifactor*10);
            icur = (n / ifactor) % 10;
            ilow = n - (n / ifactor) * ifactor;

            switch(icur){
                case 0 :
                    icout += ihight*ifactor;
                    break;
                case 1 :
                    icout += ihight*ifactor + ilow + 1;
                    break;
                default:
                    // icur > 1
                    icout += (ihight+1)*ifactor;

            }

            ifactor *= 10; // scan from the low-order to high-order
        }

        return icout;
    }
};

int main()
{
    return 0;
}

整数中1出现的次数(从1到n的整数中1出现的次数)的更多相关文章

  1. 题目1373:整数中1出现的次数(从1到n整数中1出现的次数)

    题目1373:整数中1出现的次数(从1到n整数中1出现的次数) 题目描述: 亲们!!我们的外国友人YZ这几天总是睡不好,初中奥数里有一个题目一直困扰着他,特此他向JOBDU发来求助信,希望亲们能帮帮他 ...

  2. 输入一个整数n,求从1到n这n个整数的十进制表示中1出现的次数

    题目:输入一个整数n,求从1到n这n个整数的十进制表示中1出现的次数.例如输入12,从1到12这些整数中包含1 的数字有1,10,11和12,1一共出现了5次. 分析:首先最先想到的是遍历从1到n的每 ...

  3. 算法: 整数中1出现的次数(从1到n整数中1出现的次数)

    问题: 整数中1出现的次数(从1到n整数中1出现的次数) 问题:求出1~13的整数中1出现的次数,并算出100~1300的整数中1出现的次数? 为此他特别数了一下1~13中包含1的数字有1.10.11 ...

  4. 《剑指offer》— JavaScript(31)整数中1出现的次数(从1到n整数中1出现的次数)

    整数中1出现的次数(从1到n整数中1出现的次数) 题目描述 求出1~13的整数中1出现的次数,并算出100~1300的整数中1出现的次数?为此他特别数了一下1~13中包含1的数字有1.10.11.12 ...

  5. 剑指Offer - 九度1373 - 整数中1出现的次数(从1到n整数中1出现的次数)

    剑指Offer - 九度1373 - 整数中1出现的次数(从1到n整数中1出现的次数)2014-02-05 23:03 题目描述: 亲们!!我们的外国友人YZ这几天总是睡不好,初中奥数里有一个题目一直 ...

  6. 剑指offer-31:整数中1出现的次数(从1到n整数中1出现的次数)

    参考: https://troywu0.gitbooks.io/interview/整数中出现1的次数(从1到n整数中1出现的次数).html 题目描述 求出1~13的整数中1出现的次数,并算出100 ...

  7. 剑指Offer(三十一):整数中1出现的次数(从1到n整数中1出现的次数)

    剑指Offer(三十一):整数中1出现的次数(从1到n整数中1出现的次数) 搜索微信公众号:'AI-ming3526'或者'计算机视觉这件小事' 获取更多算法.机器学习干货 csdn:https:// ...

  8. 请输入一个大于7的整数,输出小于k并且至少满足下面2个条件中的1个条件的所有正整数

    import java.util.Scanner; /** * @author:(LiberHome) * @date:Created in 2019/3/6 22:06 * @description ...

  9. 刷题之给定一个整数数组 nums 和一个目标值 taget,请你在该数组中找出和为目标值的那 两个 整数

    今天下午,看了一会github,想刷个题呢,就翻出来了刷点题提高自己的实际中的解决问题的能力,在面试的过程中,我们发现,其实很多时候,面试官 给我们的题,其实也是有一定的随机性的,所以我们要多刷更多的 ...

  10. JavaScript中一些你不一定知道的问题(持续更新中。。。。)

    一些js的问题与解析 1) ["1","2","3"].map(parseInt);的运行结果是? A.["1",&qu ...

随机推荐

  1. 【Kafka】操作命令

    生产者 ./kafka-console-producer.sh --broker-list --topic norm 消费者 ./kafka-console-consumer.sh --zookeep ...

  2. poj1067威佐夫博奕

    取石子游戏 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 31490   Accepted: 10374 Descripti ...

  3. 【笔记】Kali linux的安装 和 一些使用前的准备工作(原创+转载)

    该博文只记录笔者的蛇皮使用经历,纯新手= =,可能借鉴意义也可能没有(T _ T),侵删. 目录 kali linux 在个人计算机和在VirtualBox下的安装 kali linux 使用前准备工 ...

  4. Python自学笔记-递归函数(来自廖雪峰的官网Python3)

    感觉廖雪峰的官网http://www.liaoxuefeng.com/里面的教程不错,所以学习一下,把需要复习的摘抄一下. 以下内容主要为了自己复习用,详细内容请登录廖雪峰的官网查看.   递归函数 ...

  5. 【转】python os.popen 超时问题

    python 版本 2.5.4 (在高版本python中提倡使用 subprocess.Popen 取代 os.popen) os.popen 会出现过长时间等待导致阻塞问题, 解决方法如下: [py ...

  6. 常用硬件设备GUID

    Class GUID Device Description CDROM 4D36E965-E325-11CE-BFC1-08002BE10318 CD/DVD/Blu-ray drives DiskD ...

  7. java web 学习笔记 编码问题总结

       java web 学习笔记 编码问题总结 1.非form表单中提交的中文参数---------------------------传递给Servlet服务器时,默认以iso-8859-1解码 ...

  8. WPF 快捷键读写txt

    因为git提交需要写这次做的,所以我想弄个东西来帮我写 WPF可以使用快捷键,快捷键主要使用InputBindings,WPF读写文件很简单 我每次都要写 第一句标题 之后就是我写的修改 然后加上我名 ...

  9. 图论 Warshall 和Floyd 矩阵传递闭包

    首先我们先说下图论,一般图存储可以使用邻接矩阵,或邻接表,一般使用邻接矩阵在稠密图比较省空间. 我们来说下有向图,一般的有向图也是图,图可以分为稠密图,稀疏图,那么从意思上,稠密图就是点的边比较多,稀 ...

  10. Android基础知识笔记01—框架结构与四大组件

    -----------Andriod 01--------------->>> Andriod系统架构    linux内核与驱动层. 系统运行库层. 应用框架层. 应用层 内核驱动 ...