字符串系列---string-to-integer-atoi

题目

输入一个由数字组成的字符串,把它转换成整数并输出。例如:输入字符串”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;
    }

参考文献:http://www.treelib.com/Book-detail-id-76-aid-3901.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值