求出任意非负整数区间中1出现的次数

简介: 求出任意非负整数区间中1出现的次数

引言


最近想中有一个 类似标题的需求,看到需求的时候觉得非常简单,立马闪现出下面代码思路:

 public static int couts(int n) {
        int cout = 0;
        for (int i = 1; i <= n; i++) {
            String s = String.valueOf(i);
            String replace = s.replace("1", "");
            int i1 = s.length() - replace.length();
            cout = cout + i1;
        }
        return cout;
    }

多么简单,就是这么简单。写出来以后,稍微一过脑子,觉得不行,时间复杂度太高。所以,我们需要分析一下数据。


首先分析规律

F(3)=1
F(13)=2+4=6
F(19)=2+10=12
F(23)=3+10
F(33)=4+10
F(93)=10+10=20
F(123)=24+20+13=57


设N = abcde ,其中abcde分别为十进制中各位上的数字。


如果要计算百位上1出现的次数,它要受到3方面的影响:百位上的数字,百位一下(低位)上的数字,百位一上(高位)上的数字。


如果百位上数字为0,百位上可能出现1的次数由更高位决定。比 如:12013,则可以知道百位出现1的情况可能 是:100~199,1100~1199,2100~2199,,.........,11100~11199,一共1200个。可以看出是由更高位数字 (12)决定,并且等于更高位数字(12)乘以 当前位数(100)。


如果百位上数字为1,百位上可能出现1的次数不仅受更高位影响还受低 位影响。比如:12113,则可以知道百位受高位影响出现的情况 是:100~199,1100~1199,2100~2199,,.........,11100~11199,一共1200个。和上面情况一样,并且等 于更高位数字(12)乘以 当前位数(100)。但同时它还受低位影响,百位出现1的情况是:12100~12113,一共14个,等于低位数字(13)+1。


如果百位上数字大于1(2~9),则百位上出现1的情况仅由更高位决 定,比如12213,则百位出现1的情况 是:100~199,1100~1199,2100~2199,...........,11100~11199,12100~12199,一共有 1300个,并且等于更高位数字+1(12+1)乘以当前位数(100)。


所以代码如下:

 public static int oneCount(int n) {
        int count = 0;//1的个数
        int i = 1;//当前位
        int current = 0, after = 0, before = 0;
        while ((n / i) != 0) {
            //当前位数值
            current = (n / i) % 10;
            //高位数值
            before = n / (i * 10);
            //低位数值
            after = n - (n / i) * i; 
            //如果为0,出现1的次数由高位决定,等于高位数字 * 当前位数
            if (current == 0)
                count += before * i;
                //如果为1,出现1的次数由高位和低位决定,高位*当前位+低位+1
            else if (current == 1)
                count += before * i + after + 1;
                //如果大于1,出现1的次数由高位决定,//(高位数字+1)* 当前位数
            else {
                count += (before + 1) * i;
            }
            //前移一位
            i = i * 10;
        }
        return count;
    }

测试运行,时间耗费立马减少了很多。

目录
相关文章
|
6月前
给定 n 个整数,求里面出现次数最多的数,如果有多个重复出现的数,求值最大的那个 给定n个整数,求里面出现次数最多的数,如果有多个重复出现的数,求出值最大的一
给定 n 个整数,求里面出现次数最多的数,如果有多个重复出现的数,求值最大的那个 给定n个整数,求里面出现次数最多的数,如果有多个重复出现的数,求出值最大的一
|
6月前
|
Java
如何判断科学计数法3.14E+308 在区间3.14E+38和 3.14E+1308内
对于非常大的科学计数法表示的数值,直接将其转换为 `double` 类型可能会导致溢出。Java 中的 `double` 类型表示的最大值约为 `1.7976931348623157E+308`,因此 `3.14E+308` 已经超出了其表示范围。如果需要处理超出 `double` 类型表示范围的数值,可以使用 `BigDecimal` 类来处理。 以下是一个示例,展示如何使用 `BigDecimal` 类来比较科学计数法表示的数值是否在指定区间内: ```java import java.math.BigDecimal; public class ScientificNotationC
|
自然语言处理 算法 Python
利用函数求出一个数组最大三个数的乘积
利用函数求出一个数组最大三个数的乘积
115 0
求整数序列中出现次数最多的数
求整数序列中出现次数最多的数
164 0
随即输入10个数,并求10个整数最大值
随即输入10个数,并求10个整数最大值
99 0
随即输入10个数,并求10个整数最大值
|
机器学习/深度学习 存储 算法
【简单算法】1.两数之和,给定整数数组和目标值,找出数组中2数之和等于目标值的元素
【简单算法】1.两数之和,给定整数数组和目标值,找出数组中2数之和等于目标值的元素
【简单算法】1.两数之和,给定整数数组和目标值,找出数组中2数之和等于目标值的元素
给定两个整数,M,N,表示区间【 2……n,2 m) ,请求出这个区间有多少个满足 i%7=1
给定两个整数,M,N,表示区间【 2……n,2 m) ,请求出这个区间有多少个满足 i%7=1
89 0
给你一组数,求出其中两两最大公约数中最大的值
给你一组数,求出其中两两最大公约数中最大的值
64 0
统计正数和负数的个数然后计算这些数的平均值(循环、数组解法)
统计正数和负数的个数然后计算这些数的平均值(循环、数组解法)
204 0