3、统计数字

问题描述:

计算数字k在0到n中的出现的次数,k可能是0~9的一个值

样例

例如n=12,k=1,在 [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12],我们发现1出现了5次 (1, 10, 11, 12)

显然用暴力搜索能可以解决的,一个数字一个数字地确定,这样时间复杂度就会很高了。

《编程之美》中给我们提供了一种思路:假设有一个5位数N=ABCDE,我们现在来考虑百位上出现2的次数,即:从0到ABCDE的数中,有多少个数的百位上是2。分析完它,就可以用同样的方法去计算个位,十位,千位,万位等各个位上出现2的次数。 遵循这个思路有如下思考:(以百位出现2为例)

思路:
当k不为0时
从1至10,在它们的个位数中,任意k都出现了1次。
从1至100,在它们的十位数中,任意k都出现了10次。
从1至1000,在它们的百位数中,任意k都出现了100次。
依次类推,从1至10的i次幂,在它们的左数第二位中,任意k都出现了10的i-1次幂次。
举个例子,n = 2593,k = 5,从 1 至 2593 中,数字 5 总计出现了 813 次,其中有 259 次出现在个位,260 次出现在十位,294 次出现在百位,0 次出现在千位。
现在依次分析这些数据,首先是个位。从 1 至 2590 中,包含了 259 个 10,因此任意的 k 都出现了 259 次。最后剩余的三个数 2591, 2592 和 2593,因为它们最大的个位数字 3 < k,因此不会包含任何 5。
然后是十位。从 1 至 2500 中,包含了 25 个 100,因此任意的 k 都出现了 25×10=250 次。剩下的数字是从 2501 至 2593,它们最大的十位数字 9 > k,因此会包含全部 10 个 5。最后总计 250 + 10 = 260。
接下来是百位。从 1 至 2000 中,包含了 2 个 1000,因此任意的 k 都出现了 2×100=200次。剩下的数字是从 2001 至 2593,它们最大的百位数字 5 == k,这时情况就略微复杂,它们的百位肯定是包含 5 的,但不会包含全部 100 个。如果把百位是 5 的数字列出来,是从 2500 至 2593,数字的个数与百位和十位数字相关,是 93+1 = 94。最后总计 200 + 94 = 294。
最后是千位。现在已经没有更高位,因此直接看最大的千位数字 2 < k,所以不会包含任何 5。到此为止,已经计算出全部数字 5 的出现次数。
总结一下以上的 算法,可以看到,当计算右数第 i位包含 k 的个数时:
1.取第i位左边的数字,乘以10的i-1次幂,得到基础值a。
2.取第i位数次,计算修正值:
如果大于k,则结果为a + 10的i-1次幂;
如果小于k,则结果为a;
如果等于k,则取第i位右边数字,设为b,最后结果为a+b+1。
当k = 0 时

因为最高位不可能为0 ,所以累加到左起第二位就要结束。其次是,第i位的基础值不是最高位乘以10的i-1次幂,而是乘以10的i-1次幂减1.

这样一分析,思路就会很清晰,时间复杂度也降为O(1)。

下面是测试代码:

#include <iostream>
#include <vector>
using namespace std;
int digitCounts(int k, int n) {
int count = 0, x;
if (k == 0 && n == 0) 
count = 1;
for (int i = 1; x = n / i; i *= 10) 
{
int high = x / 10;
if (k == 0) {
if (high)
high--;
else {
count++;
break;
}
}
count += high * i;
int current = x % 10;
if (current > k) 
count += i;
else if (current == k) 
count += n - x * i + 1;
}
return count;
}

int main()
{
cout << digitCounts(1, 111) << endl;
system("pause");
return 0;
}

欢迎指正~


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 题目描述: 给定一个字符串,统计其中数字字符和空格的个数。 输入格式: 一个字符串,长度不超过 10000。 输出格式: 在一行中输出数字字符的个数和空格的个数,中间用一个空格隔开。 样例输入: 1qazxsw23 edcvfr45tgbn hy67uj m,ki89ol.\\/;p-=\\][ 样例输出: 10 6 解题思路: 遍历字符串,对于每个字符,判断是否是数字字符或者空格,如果是则计数器加一。 代码实现: ### 回答2: 题目要求统计数字字符和空格的个数。首先,我们需要读入一个字符串。然后,逐个检查字符串中的每个字符,判断其是否为数字字符或空格,并计数。最后输出结果。 代码如下所示: ```python string = input() # 读入一个字符串 num_count = 0 # 数字字符个数计数器 space_count = 0 # 空格个数计数器 # 遍历字符串中的每个字符 for char in string: if char.isdigit(): # 判断是否为数字字符 num_count += 1 # 数字字符计数器加一 elif char.isspace(): # 判断是否为空格 space_count += 1 # 空格计数器加一 print("数字字符个数:", num_count) print("空格个数:", space_count) ``` 这段代码能够实现对输入字符串的遍历,并通过判断字符的属性(是否为数字字符或空格)对相应计数器进行累加。最后,输出结果。 ### 回答3: 这是一个题目要求统计给定字符串中数字字符和空格的个数的问题。可以采用遍历字符串的方式,逐个判断每个字符是不是数字字符或者空格,然后累加计数。 首先,创建两个变量,一个用来保存数字字符的个数,一个用来保存空格的个数,初始值都为0。 然后,开始遍历给定的字符串。对于每个字符,判断它是不是数字字符。可以用ord()函数将字符转换为ASCII码,然后判断ASCII码是否在数字字符的范围内。如果是数字字符,将数字字符的个数加1;如果不是数字字符,继续判断它是不是空格字符。可以使用isspace()函数判断字符是否是空格字符。如果是空格字符,将空格字符的个数加1。 最后,输出统计结果。将数字字符的个数和空格字符的个数分别输出。 下面是代码示例: ```python def count_digits_and_spaces(string): digit_count = 0 space_count = 0 for char in string: if ord('0') <= ord(char) <= ord('9'): digit_count += 1 elif char.isspace(): space_count += 1 return digit_count, space_count string = input("请输入字符串:") digit_count, space_count = count_digits_and_spaces(string) print("数字字符的个数为:", digit_count) print("空格字符的个数为:", space_count) ``` 这样,就可以统计字符串中数字字符和空格的个数了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值