面试题12:打印1到最大的n位数##

剑指offer题目12,题目如下

输入数字n,按顺序打印出1到最大的n位十进制数,比如输入3,则打印出1,2,3一直到最大的三位数999

方法一###

和面试题11《数值的整数次方》类似,乍看觉得都是简单题目,找出最大的n位数,开始逐个打印,写出代码

public static void method_1(int n )
{
int num = 1;
int i = 1;
while(i <= n)
{
num *= 10
i++;
}
for(i = 1;i < num; i++)
System.out.print(i +" ");
}

n的范围小还好,若n的取值很大,即需要考虑大数问题,这就是一个陷阱。正确的做法是使用字符串模拟数字的加法

方法二###

字符串模拟数字加法需要建立一个字符数组,每一个都存储'0'到'9'某一个字符,首先把字符数组中每一位初始化 ‘0’,然后每一次为字符数组做+1操作,最后打印出来。其中的核心操作是

  • 1.为字符数组做+1操作
  • 2.打印数组

    代码如下:
public static void method_2(int n)
{
if(n <= 0)
return;
char num[] = new char[n];
for(int i = 0; i < num.length; i++)
num[i] = '0';
int times = 0; //
while(!increment(num))//数组+1
{
print(num); //输出数组
times ++;
if(time % 10 == 0)//10个换行
{
System.out.println();
times = 0;
}
}
}

剑指offer原书中给出数组+1的方法increment很巧妙,一种方法是字符串数组与最大的N位数99…99匹配,相等则终止递增,但时间复杂度为O(n),另一种方法是在最高位(最左边位)发生了进位,返回True,则就达到了最大的n位数,时间复杂度为O(1),代码如下

///模拟整数加法
public static boolean increment(char num[])
{
int remider = 0; //进位,5+7 = 12,此时remider=1
for(int i =num.length - 1; i >= 0;i--)
{
int count = num[i] - '0' + remider;
if(i == num.length - 1) //最后一位发生变化
count += 1;
if(count >= 10)
{
if(i == 0)
return true; //最高位发生进位
else
{
count -= 10;
remider = 1;
num[i] = (char) ('0' + count);
}
}
else
{
num[i] = (char) ('0' + count);
break;
}
}
return false;
}

接下开考虑如何打印数字,由于数字由字符数组表示,位数不足n的前面用0补位,比如 当输入的n=3时,数字66的数组表示方式是066,因此在输出的时候遇到前面首数字前面为0的应该跳过去不打印

public static void print(char num[])
{
for(int i = 0;i <= num.length - 1;i++)
{
if(num[i] != '0')
{
for(int j = i;j <= num.length - 1;j++)
{
System.out.print(num[j]);
}
System.out.print(" ");
break;
}
}
}

结果:

方法三###

先给出一个问题,求0~100中数字7出现的个数

可以把数字化成这样00,01,02...98,99,每一数位都是由0到9排列一遍,总共出现100 * 2 = 200个数字,由于0~9这10个数出现的次数相等,则7出现了 200/10 = 20次。

同理,可以在所有的数字前面补0,就会得到n位数字,每一位都是0~9的全排列,只是排在数字前面的0我们不打印出来,由于每一位都是0-9的全排列,可以想到使用递归思路,求数字的全排列Permutation也是类似思想,但有少许区别

// 使用递归
public static void method_3(int n)
{
if(n <= 0)
return;
char num[] = new char[n];
for(int i = 0; i < 10; i++)
{
num[0] = (char) (i + '0');//第0位 全排列0~9
recursive(num,n ,0);
}
}
public static void recursive(char num[],int length ,int index) //index全排列到第几位了
{
if(index == length - 1)
{
System.out.println();
print(num);
return;
}
for(int i = 0; i < 10; i++)
{
num[index + 1] = (char) (i + '0');
recursive(num,length,index + 1);
}
}

总结###

题目简单么?这可要比这些天做的各大公司笔试题算法题目简单太多了,有些公司出的题需要看半天才能表达的意思。这么容易理解的题目如果真是bug free 的做出来还真是有些难度,本节两个关键点:

  • 字符串模拟整数的加法
  • 递归方法全排列数字

《剑指offer》面试题12:打印1到最大的n位数的更多相关文章

  1. 面试题12:打印1到最大的n位数

    // 面试题12_打印1到最大的n位数.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include <iostream> ...

  2. 剑指offer编程题Java实现——面试题12打印1到最大的n位数

    题目:打印1到最大的n位数 输入数字n,按顺序打印输出从1到最大的n位十进制数,比如输入3,打印从1到999. 这道题考察的地方是如何表示大数问题.由于n是任意大的数组,如果n太大的话n位数就超过了l ...

  3. 剑指Offer:面试题12——打印1到最大的n位数(java实现)

    问题描述: 输入数字n,按顺序打印出从1到最大的n位十进制数,比如输入3,则打印出1,2,3一直到最大的3位数即999. 思路1:最简单的想法就是先找出最大的n位数,然后循环打印即可. public ...

  4. 《剑指offer》面试题12 打印1到最大的n位数 Java版

    书中方法:这道题的一个陷阱在于不能用int或者long去存储你要打印的数,然后用打印函数打印,因为这个数可能会很大.如果加1后超出了最大的n位数,就不打印了.用最高位是否进位判断是否结束,打印的时候注 ...

  5. 剑指offer-面试题12.打印1到最大的n位数

    题目:输入数字n,按照打印出从1最大的n位10进制数.比如3,则 打印出1.2.3一直到最大的3位数即999 1.你觉得如果面试会有这么简单的题,那 只能说明你---太天真. 2.n=3尚可,如果n= ...

  6. 题目12 打印1到最大的n位数

    ///////////////////////////////////////////////////////////////////////////////////// // 2.打印1到最大的n位 ...

  7. 12 打印1到最大的n位数

    输入数字 n,按顺序打印出从 1 最大的 n 位十进制数.比如输入 3,则打印出 1.2.3 一直到最大的 3 位数即 999.由于 n 可能会非常大,因此不能直接用 int 表示数字,而是用 cha ...

  8. 【面试题012】打印1到最大的n位数

    [面试题012]打印1到最大的n位数  大数问题 字符串中的每一个字符都是‘0’到‘9’之间的某一个字符,用来表示数字中的一位,因为数字最大是n位的,因此我们需要一个长度为n+1的字符串,字符串的最后 ...

  9. 剑指offer编程题Java实现——面试题12相关题大数的加法、减法、乘法问题的实现

    用字符串或者数组表示大数是一种很简单有效的表示方式.在打印1到最大的n为数的问题上采用的是使用数组表示大数的方式.在相关题实现任意两个整数的加法.减法.乘法的实现中,采用字符串对大数进行表示,不过在具 ...

随机推荐

  1. jquery mobile 的优缺点

    jQuery Mobile 优点 跨浏览器兼容性最好,几乎兼容所有的平台和浏览器: 入门简单,语法简洁,编码灵活,一些简单的应用直接用HTML既可实现,无需Javascript: 开源插件与第三方扩展 ...

  2. python 编码

    python相关:1. 设置编码:import sysreload(sys)sys.setdefaultencoding('utf-8') 开头设置:  # -*- coding: utf-8 -*- ...

  3. 使用Volley执行网络数据传输

    首先需要实例化一个RequestQueue RequestQueue queue = Volley.newRequestQueue(this); 然后是根据提供的URL请求字符串响应 String u ...

  4. 导出Excel

    一.asp.net中导出Execl的方法:在asp.net中导出Execl有两种方法,一种是将导出的文件存放在服务器某个文件夹下面,然后将文件地址输出在浏览器上:一种是将文件直接将文件输出流写给浏览器 ...

  5. JDK安装与环境变量配置

    1.安装JDK 选择安装目录 安装过程中会出现两次 安装提示 .第一次是安装 jdk ,第二次是安装 jre .建议两个都安装在同一个java文件夹中的不同文件夹中.(不能都安装在java文件夹的根目 ...

  6. ubuntu13.04下建立嵌入式开发平台

    linux下建立嵌入式交叉开发平台,需要三个基本部分:编辑工具.交叉工具链以及平台相关库. 一.编辑工具: 一般Linux系统本身都带有编辑工具,比如VI.VIM.gedit等.这里记录的是第三方编辑 ...

  7. Adobe Reader/Acrobat修改页面底色为豆沙绿

    Adobe Acrobat_Pro_8修改PDF页面底色为豆沙绿保护视力(同样适用于Adobe Reader) http://jingyan.baidu.com/article/9989c746189 ...

  8. SQL Server 在数据库中查找字符串(不知道表名的情况下 查找字符串)

    declare @key varchar(30)set @key = '广州' --替换为要查找的字符串DECLARE @tabName VARCHAR(40),@colName VARCHAR(40 ...

  9. Python开发入门与实战20-微信开发配置

    随着移动互联网时代的来临,微信已经成为移动互联网移动端的主要入口,现在很多的大型企业都有自己的微信服务号,如:银行业有自己的微银行,基金公司的公众服务号.通过微信入口可以方便快速的实现企业提供的服务. ...

  10. SQL Server 简介

    什么是数据库? 数据库(Database)是按照数据结构来组织.存储和管理数据的仓库.是以一定方式储存在一起.能为多个用户共享.具有尽可能小的冗余度的特点.是与应用程序彼此独立的数据集合. SQL S ...