题目
输入一个由数字组成的字符串,把它转换成整数并输出。例如:输入字符串”123”,输出整数123。
给定函数原型int StrToInt(const char *str) ,实现字符串转换成整数的功能,不能使用库函数atoi。
分析
基本思路便是:从左至右扫描字符串,把之前得到的数字乘以10,再加上当前字符表示的数字:
int StrToInt(const char *str)
{
while (*str != 0)
{
int c = *str - '0';
n = n * 10 + c;
++str;
}
return n;
}
上述代码忽略了以下细节:
1)空指针输入:输入的是指针,在访问空指针时程序会崩溃,因此在使用指针之前需要先判断指针是否为空。
2)正负符号:整数不仅包含数字,还有可能是以’+’或’-‘开头表示正负整数,因此如果第一个字符是’-‘号,则要把得到的整数转换成负整数。
3)非法字符:输入的字符串中可能含有不是数字的字符。因此,每当碰到这些非法的字符,程序应停止转换。
4)整型溢出:输入的数字是以字符串的形式输入,因此输入一个很长的字符串将可能导致溢出。
溢出问题:
一般说来,当发生溢出时,取最大或最小的int值。即大于正整数能表示的范围时返回MAX_INT:2147483647;小于负整数能表示的范围时返回MIN_INT:-2147483648。
sign用来处理数字的正负,当为正时sign > 0,当为负时sign < 0
n存放最终转换后的结果
c表示当前数字
你可能会编写如下代码段处理溢出问题:
//当发生正溢出时,返回INT_MAX
if ((sign == '+') && (c > MAX_INT - n * 10))
{
n = MAX_INT;
break;
}
//发生负溢出时,返回INT_MIN
else if ((sign == '-') && (c - 1 > MAX_INT - n * 10))
{
n = MIN_INT;
break;
}
上述代码会出错:n*10可能超出范围
处理溢出方法
1)把转换后返回的值n定义成long long,即long long n
2)另外一种则是只比较n和MAX_INT / 10的大小,即:
若n > MAX_INT / 10,那么说明最后一步转换时,n*10必定大于MAX_INT,所以在得知n > MAX_INT / 10时,当即返回MAX_INT。
若n == MAX_INT / 10时,那么比较最后一个数字c跟MAX_INT % 10的大小,即如果n == MAX_INT / 10且c > MAX_INT % 10,则照样返回MAX_INT。
所以处理溢出代码:
c = *str - '0';
if (sign > 0 && (n > MAX_INT / 10 || (n == MAX_INT / 10 && c > 7)))
{
break;
}
else if (sign < 0 && (n > (unsigned)MIN_INT / 10 ||(n==unsigned)MIN_INT / 10 && c > 8))
{
n = MIN_INT;
break;
}
实现
public int atoi(String str){
if(str==null || str.length()==0){
return 0;
}
//处理空格
int index=0;
while(index<str.length() && str.charAt(index)==' '){
index++;
}
//处理符号,1 -1
int sign=1;
if(str.charAt(index)=='+' || str.charAt(index)=='-'){
if(str.charAt(index)=='-'){
sign=-1;
}
index++;
}
int n=0;
while(index<str.length() && str.charAt(index)>='0' && str.charAt(index)<='9'){
int c=str.charAt(index)-'0';
if(sign>0 && (n>Integer.MAX_VALUE/10 || (n==Integer.MAX_VALUE/10 && c>Integer.MAX_VALUE%10))){
n=Integer.MAX_VALUE;
break;
}
if(sign<0 && (n>Integer.MAX_VALUE/10 || (n==Integer.MAX_VALUE/10 && c>8))){
n=Integer.MIN_VALUE;
break;
}
n=n*10+c;
index++;
}
return n*sign;
}