题目:

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

思路:

比如求1到34597中1出现的次数。首先划分两段:1到4597,4598到34597.
第二段的话,最高位1出现的次数为10000~19999,总共10000次(假设n的位数为b,即10的b次方个);另一部分是其他位的1。
对于求其他位的1,将第二段再次划分:4598到14597,14598到24597,24598到34597.对于每一段,选定一位为1时,其余位置可以为0~9中任意一个,则每段有10X10X10X3个;总共有10X10X10X3X3个。即10的(b-2)次方乘以(b-2)乘以最高位数字。
这样,整个第二段4598到34597的个数求得,对于第一段用递归方法求即可。

总结,可以将1到n划分为这样几步:(假设n的位数为len,最高位数字为first,n也记为abxxx...)

A. 从1到bxxx... 递归求解

B.a 从bxxx...+1到abxxx...的最高位1个数:如果first>1,则为10的first次方;否则为bxxx...+1个。

B.b 从bxxx...+1到abxxx...的除最高位有1外,其他位1的个数:10的(len-2)次方乘以(len-1)乘以first个。

注意:

实现的时候,可以用字符串来处理数字n。这样去掉最高位后的数字,只需要将字符指针加1即可。

需要注意的输入:长度为1,且first为0;长度为1,且first>0。(因为后面有len-2)

代码:

class Solution {
public:
int NumberOf1Between1AndN_Solution(int n)
{
if(n<=) return ; char s[];
sprintf(s,"%d",n); return NumberOf1(s);
}
private:
int NumberOf1(char *s)
{
if(s==NULL || *s<'' || *s>'' || *s=='\0') return ; int len=strlen(s);
int first=s[]-'';//最高位数字 if(len==1 && first==0) return 0;
if(len==1 && first>0) return 1; int numFirstDigit=;//最高位的1的个数
if(first>) numFirstDigit=PowerOf10(len-);
else if(first==)
numFirstDigit=atoi(s+1)+; int numOtherDigit=;//其他位的1的个数
if(len>=)
numOtherDigit=PowerOf10(len-)*first*(len-1); int numRecursive=NumberOf1(s+);//递归求解的1的个数 return numFirstDigit+numOtherDigit+numRecursive;
} int PowerOf10(int n)
{
int res=;
for(int i=;i<n;++i)
{
res=res*;
}
return res;
}
};

讨论:

在B.b的求个数过程中,是不是有重复的,比如选定第二位为1时,其他位可以从0到9任选,而选定第三位为1时,其他为又是从0到9任选。这样,对于形如”第二位为1、第三位为1“的数字,则重复计算了。

同样,B.b和B.a的计算中,好像也是有重复计算了的。

是我理解错了吗?求讨论。

答:我理解错了,并没有重复计数。

【剑指offer】面试题32:从1到n整数中1出现的次数的更多相关文章

  1. 剑指Offer:面试题32——从1到n整数中1出现的次数(java实现)

    问题描述: 输入一个整数n,求1到n这n个整数的十进制表示中1出现的次数.例如输入12,从1到12这些整数中包含1的数字有1,10,11,12,1一共出现了5次. 思路:(不考虑时间效率的解法,肯定不 ...

  2. 【剑指Offer面试编程题】题目1373:整数中1出现的次数--九度OJ

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

  3. 剑指Offer的学习笔记(C#篇)-- 整数中1出现的次数(从1到n整数中1出现的次数)

    题目描述 求出1~13的整数中1出现的次数,并算出100~1300的整数中1出现的次数?为此他特别数了一下1~13中包含1的数字有1.10.11.12.13因此共出现6次,但是对于后面问题他就没辙了. ...

  4. 【剑指Offer】31、从1到n整数中1出现的次数

      题目描述:   求出1~13的整数中1出现的次数,并算出100~1300的整数中1出现的次数?为此他特别数了一下1~13中包含1的数字有1.10.11.12.13因此共出现6次,但是对于后面问题他 ...

  5. 【剑指Offer面试题】 九度OJ1517:链表中倒数第k个结点

    鲁棒性是指程序可以推断输入是否符合规范要求,并对不和要求的输入予以 合理的处理. 题目链接地址: http://ac.jobdu.com/problem.php?pid=1517 题目1517:链表中 ...

  6. 剑指offer——面试题32.1:分行从上到下打印二叉树

    void BFSLayer(BinaryTreeNode* pRoot) { if(pRoot==nullptr) return; queue<BinaryTreeNode*> pNode ...

  7. 剑指offer——面试题32:从上到下打印二叉树

    void BFS(BinaryTreeNode* pRoot) { if(pRoot==nullptr) { cout<<"empty binary tree!"< ...

  8. 【剑指offer 面试题38】数字在排序数组中出现的次数

    思路: 利用二分查找,分别查找待统计数字的头和尾的下标,最后做差加一即为结果. C++: #include <iostream> #include <vector> using ...

  9. 剑指offer——面试题15.2:判断两个整数m和n的二进制中相差多少位

    #include"iostream" using namespace std; int CountDifferentBit(int m,int n) { ,diff=m^n; wh ...

  10. 面试题32.从1到n整数中1出现的次数

    题目:输入一个整数n,求从1到n这n个整数的十进制表示中1出现的次数.例如输入12,从 1到12这些整数中包含1的数字中1,10,11和12,1一共出现了5次 本题可以直接变量1到n的n个数然后分别计 ...

随机推荐

  1. 源码分析之struts1自定义方法的使用与执行过程

    最近有人问我,你做项目中用户的一个请求是怎么与struts1交互的,我说请求的url中包含了action的名字和方法名,这样就可以找到相应方法,执行并返回给用户了. 他又问,那struts1中有什么方 ...

  2. python glob标准库基础学习

    #glob文件名模式匹配#作用:使用unix shell规则查找与一个模式匹配文件名"""尽管glob api很小,但这个模块很强大,只要程序需要查找文件系统中名字与某种 ...

  3. Laravel-高级篇-Auth-数据迁移-数据填充

    (慕课网_轻松学会Laravel-高级篇_天秤vs永恒老师_http://www.imooc.com/learn/702) 一.生成Auth所需文件 在Artisan控制台输入以下命令 php art ...

  4. HDOJ/HDU 2544 最短路---dijkstra算法

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2544 这题的思路可以见这里(同一类型):http://blog.csdn.net/xiaozhuaix ...

  5. 画8_hdu_1256(图形).java

    画8 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submiss ...

  6. [Redux] Extracting Presentational Components -- Footer, FilterLink

    Code to be refactored: let nextTodoId = 0; class TodoApp extends Component { render() { const { todo ...

  7. Dalvik虚拟机Java堆创建过程分析

    文章转载至罗升阳CSDN社区博客,原地址: http://blog.csdn.net/luoshengyang/article/details/6557518 近年来,手机移动平台越来越火爆.打开自己 ...

  8. c++11 生产者/消费者

    下面是一个生产者消费者问题,来介绍condition_variable的用法.当线程间的共享数据发生变化的时候,可以通过condition_variable来通知其他的线程.消费者wait 直到生产者 ...

  9. Win7x64安装Oracle11201x64 解决PLSQL Developer无法找到oci问题

    http://blog.sina.com.cn/s/blog_4c7628c40101cf56.html http://blog.csdn.net/shenkxiao/article/details/ ...

  10. 注册DLL,Unregister DLL

    前一篇文章,反复注册,反注册.... 写了一个小工具 怎么传图片感觉不对劲,StatusBar应改成 拖动DLL至上方 文件 http://files.cnblogs.com/magicdawn/Dl ...