44_DigitsInSequenc

package pers.lyt.java;

//题目 
//	数字以0123456789101112131415…的格式序列化到一个字符序列中。在这个序列中,第5位
//(从0开始计数)是5,第13位是1,第19位是4,等等。请写一个函数求任意位对应的数字。
//思路
//	逐一枚举数字,计算每个数字的位数相加,效率太低。
//	观察规律:
//个位数的个数一共有10个,即0~9,共占了10*1位数字;
//两位数的个数一共有90个,即10~99,每个数字占两位,共占了90*2位数字;
//……
//m位数的个数一共有9*10^(m-1)个,每个数字占m位,占了9*10^(m-1)*m位数字。
//判断第n个对的数字是属于几位数,再从几位数中进行寻找。
//注:从0开始计数,从1开始计数开始要index++
public class Offer44_DigitsInSequenc {
	public int digitAtIndex(int index) {
		if (index < 0)
			return -1;
		int m = 1;// m位数
		while (true) {
			// m位数的总位数
			int numbers = numbersOfIntegers(m);
			if (index < numbers)
				return getDigit(index, m);
			index -= numbers;
			m++;
		}
	}

	/*
	 * 返回m位数的总位数 例如,两位数一共有90个:10~99;三位数有900个:100~999
	 */
	private int numbersOfIntegers(int m) {
		if (m == 1)
			return 10;
		return (int) (9 * Math.pow(10, m - 1) * m);
	}

	/*
	 * 获取数字
	 */
	private int getDigit(int index, int m) {
		int number = getFirstNumber(m) + index / m; // 对应的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);
	}

	public static void main(String[] args) {
		Offer44_DigitsInSequenc demo = new Offer44_DigitsInSequenc();
		System.out.println(demo.digitAtIndex(0) == 0);
		System.out.println(demo.digitAtIndex(1) == 1);
		System.out.println(demo.digitAtIndex(10) == 1);
		System.out.println(demo.digitAtIndex(190) == 1);
		System.out.println(demo.digitAtIndex(1000) == 3);
		System.out.println(demo.digitAtIndex(1001) == 7);
		System.out.println(demo.digitAtIndex(1002) == 0);
	}
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值