剑指Offer对答如流系列 - 数字序列中某一位的数字

面试题44:数字序列中某一位的数字

题目描述

数字以0123456789101112131415…的格式序列化到一个字符序列中。在这个序列中,第5位(从0开始计数)是5,第13位是1,第19位是4,等等。请写一个函数求任意位对应的数字。

问题分析

这个寻求高效的解决方法,也是寻找规律:

  1. 个位数的个数一共有10个,即0~9,共占了10*1位数字;(特殊)
  2. 两位数的个数一共有90个,即10~99,每个数字占两位,共占了90*2位数字;
  3. ……
  4. m位数的个数一共有9*10^(m-1)个,每个数字占m位,占了9*10^(m-1)*m位数字。

  判断第n个对的数字是属于几位数,再从几位数中进行寻找。

举个例子:

0123456789101112131415161718192021....

我们要取第33位。

为了更形象,我们把上面的规律抽象成比较直观的场景:我们把实际的数字想成它们在一个个方格里面存储。一个方格可能储存多个数字。

首先我们要找出第33位的字符对应的数字的位数。因为个位数的个数有10个,两位数的个数有90个,33介于10-100之间,可以确定是两位。

利用一个方格存储两位的特点:

(33-10 )/2 得出11,这个数字加上10 就得出了第33位字符对应的数字是 21

(33-10)%2 得出 我们要的字符是 对应数字的第1位 即1

问题解答

  public int digitAtIndex(int index) {
        if(index<0){
            return -1;
        }
        // m位数
        int m=1;
        while(true) {
            // m位数的个数
            int numbers=numbersOfIntegers(m);
            if(index<numbers*m) {
                return getDigit(index,m);
            }
            index-=numbers*m;
            m++;
        }
    }

    // 返回m位数的总个数
    private int numbersOfIntegers(int m) {
        if(m == 1) {
            return 10;
        }
        return (int) (9*Math.pow(10, m-1));
    }

    // 获取数字
    private int getDigit(int index, int m) {
        // 对应的m位数
        int number = getFirstNumber(m)+index/m;
        // 在数字中的位置
        int indexFromRight = m - index%m;
        for(int i=1; i < indexFromRight; i++) {
            number/=10;
        }
        return number%10;
    }

    // 第一个m位数 例如第一个两位数是10,第一个三位数是100
    private int getFirstNumber(int m) {
        if(m==1) {
            return 0;
        }
        return (int) Math.pow(10, m-1);
    }

原文地址:https://www.cnblogs.com/JefferyChenXiao/p/12246507.html

时间: 2024-08-01 11:42:15

剑指Offer对答如流系列 - 数字序列中某一位的数字的相关文章

剑指offer-面试题44-数字序列中某一位的数字-脑筋急转弯

/* 题目: 数字以0123456789101112131415…的格式序列化到一个字符序列中. 在这个序列中,第5位(从0开始计数,即从第0位开始)是5,第13位是1,第19位是4,等等. 请写一个函数,求任意第n位对应的数字. */ #include<iostream> #include<string.h> using namespace std; //自己实现pow,使用codeblocks中的pow,会转化为浮点数进行截断,得到的结果不准确. int pow(int num

剑指Offer对答如流系列 - 数组中数字出现的次数

面试题56:数组中数字出现的次数 题目描述 问题(1)数组中只出现一次的两个数字 一个整型数组里除了两个数字之外,其他的数字都出现了两次.请写程序找出这两个只出现一次的数字.要求时间复杂度是O(n),空间复杂度是O(1). 问题(2)数组中唯一只出现一次的数字 在一个数组中除了一个数字只出现一次之外,其他数字都出现了三次.请找出那个只出现一次的数字. 问题分析 问题(1)分析 在这篇文章剑指Offer对答如流系列 - 二进制中 1 的个数中,我们详细探讨了位运算,其中有重要的一条:两个相同的数异

《剑指offer》第四十四题:数字序列中某一位的数字

// 面试题44:数字序列中某一位的数字 // 题目:数字以0123456789101112131415…的格式序列化到一个字符序列中.在这 // 个序列中,第5位(从0开始计数)是5,第13位是1,第19位是4,等等.请写一 // 个函数求任意位对应的数字. #include <iostream> #include <algorithm> using namespace std; int countOfIntegers(int digits); int digitAtIndex(

剑指Offer对答如流系列 - 序列化二叉树

面试题37:序列化二叉树 题目描述 请实现两个函数,分别用来序列化和反序列化二叉树. 树的结构定义如下: public class Node { int val = 0; Node left = null; Node right = null; public Node(int val) { this.val = val; } } 问题分析 一般情况下,需要采用前/后序遍历和中序遍历才能确定一个二叉树,具体的内容我们之前探讨过 剑指Offer对答如流系列 - 重建二叉树 但是采用这种方式进行序列化

剑指Offer对答如流系列 - 礼物的最大价值

面试题47:礼物的最大价值 题目描述 在一个m×n的棋盘的每一格都放有一个礼物,每个礼物都有一定的价值(价值大于0).你可以从棋盘的左上角开始拿格子里的礼物,并每次向左或者向下移动一格直到到达棋盘的右下角.给定一个棋盘及其上面的礼物,请计算你最多能拿到多少价值的礼物? 比如下面的棋盘中,如果按照红色数字的路线走可以拿到最大价值为53的礼物 问题分析 动态规划:定义f(i,j)为到达(i,j)位置格子时能拿到的礼物总和的最大值,则有:f(i,j)=max{f(i-1,j),f(i,j-1)}+va

剑指Offer对答如流系列 - 把数组排成最小的数

面试题45:把数组排成最小的数 题目描述 输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个.例如输入数组{3, 32, 321},则打印出这3个数字能排成的最小数字321323. 问题分析 之前我们做过字符全排列的习题 剑指Offer对答如流系列 - 字符串的排列,但是将算法思想应用到这一题的话,效果不好,求出所有的组合,再计算出组合的最小值,这效率该多低啊. 我们还要进一步探究,看看有没有不错的规律,供我们使用. 因为数字拼接后的长度一样,拼接后的结果

剑指Offer对答如流系列 - 从上往下打印二叉树

面试题32:从上往下打印二叉树 题目描述 树的结构定义如下: public class Node{ int e; Node left; Node right; Node(int x) { e = x; } } (一)不分行从上到下打印二叉树 从上往下打印出二叉树的每个结点,同一层的结点按照从左到右的顺序打印. 比如下面二叉树,输出顺序为 8 6 10 5 7 9 11 (二)分行从上到下打印二叉树 从上到下按层打印二叉树,同一层的结点按从左到右的顺序打印,每一层打印到一行. 比如下面二叉树,输出

剑指Offer对答如流系列 - 求1+2+…+n

面试题64:求1+2+-+n 题目描述 求1+2+-+n,要求不能使用乘除法.for.while.if.else.switch.case等关键字及条件判断语句(A?B:C). 问题分析 有了那么多限制,剩下的我们可以选择 单目运算符:++和--,双目运算符:+,-,移位运算符<>,关系运算符>,<等 逻辑运算符&&,||,&,|,^,赋值= 既然是一个等差数列,和为(n+1)*n/2 我们之前详细探讨了位运算剑指Offer对答如流系列 - 二进制中 1 的个

剑指Offer对答如流系列 - 不用加减乘除做加法

面试题65:不用加减乘除做加法 题目描述 写一个函数,求两个整数之和,要求在函数体内不得使用+.-.×.÷四则运算符号. 问题分析 我们之前详细探讨了位运算 剑指Offer对答如流系列 - 二进制中 1 的个数,已经非常非常详细了. 这道题仅仅是让做加法,我们除此之外还是做了乘除与减法. 记不清的朋友可以回头看看. 这里象征性地做一次解答吧 问题解答 public int add(int num1,int num2) { while(num2!=0){ int sum=num1^num2; in