整数中1出现的次数(从1到n的整数中1出现的次数)
题目
求出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出现的次数)的更多相关文章
- 题目1373:整数中1出现的次数(从1到n整数中1出现的次数)
题目1373:整数中1出现的次数(从1到n整数中1出现的次数) 题目描述: 亲们!!我们的外国友人YZ这几天总是睡不好,初中奥数里有一个题目一直困扰着他,特此他向JOBDU发来求助信,希望亲们能帮帮他 ...
- 输入一个整数n,求从1到n这n个整数的十进制表示中1出现的次数
题目:输入一个整数n,求从1到n这n个整数的十进制表示中1出现的次数.例如输入12,从1到12这些整数中包含1 的数字有1,10,11和12,1一共出现了5次. 分析:首先最先想到的是遍历从1到n的每 ...
- 算法: 整数中1出现的次数(从1到n整数中1出现的次数)
问题: 整数中1出现的次数(从1到n整数中1出现的次数) 问题:求出1~13的整数中1出现的次数,并算出100~1300的整数中1出现的次数? 为此他特别数了一下1~13中包含1的数字有1.10.11 ...
- 《剑指offer》— JavaScript(31)整数中1出现的次数(从1到n整数中1出现的次数)
整数中1出现的次数(从1到n整数中1出现的次数) 题目描述 求出1~13的整数中1出现的次数,并算出100~1300的整数中1出现的次数?为此他特别数了一下1~13中包含1的数字有1.10.11.12 ...
- 剑指Offer - 九度1373 - 整数中1出现的次数(从1到n整数中1出现的次数)
剑指Offer - 九度1373 - 整数中1出现的次数(从1到n整数中1出现的次数)2014-02-05 23:03 题目描述: 亲们!!我们的外国友人YZ这几天总是睡不好,初中奥数里有一个题目一直 ...
- 剑指offer-31:整数中1出现的次数(从1到n整数中1出现的次数)
参考: https://troywu0.gitbooks.io/interview/整数中出现1的次数(从1到n整数中1出现的次数).html 题目描述 求出1~13的整数中1出现的次数,并算出100 ...
- 剑指Offer(三十一):整数中1出现的次数(从1到n整数中1出现的次数)
剑指Offer(三十一):整数中1出现的次数(从1到n整数中1出现的次数) 搜索微信公众号:'AI-ming3526'或者'计算机视觉这件小事' 获取更多算法.机器学习干货 csdn:https:// ...
- 请输入一个大于7的整数,输出小于k并且至少满足下面2个条件中的1个条件的所有正整数
import java.util.Scanner; /** * @author:(LiberHome) * @date:Created in 2019/3/6 22:06 * @description ...
- 刷题之给定一个整数数组 nums 和一个目标值 taget,请你在该数组中找出和为目标值的那 两个 整数
今天下午,看了一会github,想刷个题呢,就翻出来了刷点题提高自己的实际中的解决问题的能力,在面试的过程中,我们发现,其实很多时候,面试官 给我们的题,其实也是有一定的随机性的,所以我们要多刷更多的 ...
- JavaScript中一些你不一定知道的问题(持续更新中。。。。)
一些js的问题与解析 1) ["1","2","3"].map(parseInt);的运行结果是? A.["1",&qu ...
随机推荐
- shim 和 polyfill
在前端,有两个词经常被提及:shim 和 polyfill.最近在翻译文章时又遇到了 polyfill 这个词,准备把这两个概念理清楚. 关于 JavaScript 的兼容性问题,通常有不同的解决方案 ...
- Android打包版本号设置
之前没有设置过打包的命名,每次打包都是默认的"app-realease.apk",之后手动修改名字来显示出它是一个新版本. 晚上学习了如何配置打包名称,很简单,修改build.gr ...
- 约会安排HDU - 4553
寒假来了,又到了小明和女神们约会的季节. 小明虽为屌丝级码农,但非常活跃,女神们常常在小明网上的大段发言后热情回复"呵呵",所以,小明的最爱就是和女神们约会.与此同时,也有很多基 ...
- 2017-2018 ACM-ICPC, NEERC, Southern Subregional Contest, qualification stage (Online Mirror, ACM-ICPC Rules, Teams Preferred)
题目链接:http://codeforces.com/problemset/problem/847/I I. Noise Level time limit per test 5 seconds mem ...
- Codecraft-17 and Codeforces Round #391 (Div. 1 + Div. 2, combined)D. Felicity's Big Secret Revealed
题目连接:http://codeforces.com/contest/757/problem/D D. Felicity's Big Secret Revealed time limit per te ...
- 推荐系统相关算法(1):SVD
假如要预测Zero君对一部电影M的评分,而手上只有Zero君对若干部电影的评分和风炎君对若干部电影的评分(包含M的评分).那么能预测出Zero君对M的评分吗?答案显然是能.最简单的方法就是直接将预测分 ...
- html表格宽度设置失效
问题描述: 我在写一个网页table时,table宽度超过了我预想的宽度,我想把它设置小一点,但总是没效果.改到怀疑人生!代码如下: 经过多次调试后发现一个问题,table可以改变大小,但是会有一个最 ...
- Django学习中遇到的问题(1)django migration No migrations to apply
C:\Users\Desktop\homeWork\Django_stu_man>python manage.py makemigrations Migrations for 'app01': ...
- webpack2使用ch2-entry和output简要说明
1 entry打包入口 打包字符串和数组 const webpack = require('webpack'), path = require('path'); module.exports = { ...
- HDU5661 Claris and XOR
我们求二进制是怎么求的呢:先看看二进制的每一位代表多大:.......32 16 8 4 2 1 假如n=10, ..... 32>n ,不要. 16>n,不要. 8<=n,要,然后 ...