题目:输入数字 n,按顺序输出从 1最大的 n位 10进制数。比如输入 3,则输出 1、 2、 3一直到最大的 3位数即 999。
分析:这是一道很有意思的题目。看起来很简单,其实里面却有不少的玄机。
void Print1ToMaxOfNDigits_1(int n) { // calculate 10^n int number = 1; int i = 0; while(i++ < n) number *= 10; // print from 1 to (10^n - 1) for(i = 1; i < number; ++i) printf("%d\t", i); }
初看之下,好像没有问题。但如果我们仔细分析这个问题,就能注意到这里没有规定n的范围,当我们求最大的n位数的时候,是不是有可能用整型甚至长整型都会溢出?
分析到这里,我们很自然的就想到我们需要表达一个大数。最常用的也是最容易实现的表达大数的方法是用字符串或者整型数组(当然不一定是最有效的)。
用字符串表达数字的时候,最直观的方法就是字符串里每个字符都是’0’到’9’之间的某一个字符,表示数字中的某一位。因为数字最大是n位的,因此我们需要一个n+1位字符串(最后一位为结束符号’\0’)。当实际数字不够n位的时候,在字符串的前半部分补零。这样,数字的个位永远都在字符串的末尾(除去结尾符号)。
首先我们把字符串中每一位数字都初始化为’0’。然后每一次对字符串表达的数字加1,再输出。因此我们只需要做两件事:一是在字符串表达的数字上模拟加法。另外我们要把字符串表达的数字输出。值得注意的是,当数字不够n位的时候,我们在数字的前面补零。输出的时候这些补位的0不应该输出。比如输入3的时候,那么数字98以098的形式输出,就不符合我们的习惯了。
基于上述分析,我们可以写出如下代码:
package test; import java.util.Scanner; public class Test { public static void print1toMax(int n) { if(n<=0) return ; char number[] = new char[n]; for(int i=0;i<number.length;i++) number[i]= '0'; while(!incent(number)) { print(number); } } private static void print(char[] number) { int n = number.length; boolean is_0 = true; for(int i=0;i<n;i++) { if(number[i]!='0') is_0 = false; if(!is_0) System.out.print(number[i]); } System.out.println(); } public static boolean incent(char []number) { boolean isoverflow = false; int jinwei = 0; int n= number.length; for(int i=n-1;i>=0;--i) { int num = number[i] -'0'+jinwei; if(i==n-1) num++; if(num>=10) { if(i==0) { isoverflow = true; } else { jinwei = 1; num = num - 10; number[i] = (char)(num +'0'); } } else { number[i] = (char)('0'+num); break; } } return isoverflow; } //一下是第二种方法, public static void pint1toMaxMethod2(int n) { if(n<=0) return ; char number[] = new char[n]; for(int i=0;i<10;i++) { number[0] = (char)(i+'0'); printdigui(number,n,0); } } public static void printdigui(char number[],int length,int index) { int n=number.length; if(index==length-1) { print(number); return ; } for(int i=0;i<10;i++) { number[index+1] = (char)('0'+i); printdigui(number,length,index+1); } } public static void main(String[] args) { Scanner sc = new Scanner(System.in); int n = sc.nextInt(); // print1toMax(n); System.out.println("第二种方法"); pint1toMaxMethod2(n); } }
本文已经收录到《剑指Offer——名企面试官精讲典型编程题》一书中,有改动,书中的分析讲解更加详细。欢迎关注。
博主何海涛对本博客文章享有版权。网络转载请注明出处http://zhedahht.blog.163.com/。整理出版物请和作者联系