剑指offer—第42题分析

题目:输入一个字符串,输出对应的整数。例如:输入"12345",输出对应的整数12345.

这不就是库函数里面的atoi吗!!! 大部分人都不加思索的写出如下程序。
intmy_atoi(char*arr)
{
                assert(arr);
                int num = 0;
                while(*arr)
                {
                                num = num * 10 + *arr - '0' ;
                                arr++;
                }
                return num;
}

貌似上面的程序看起来一点问题都没有,实际上问题大了!!!

分析:
1、如果输入的字符包含不是数字字符的字符???
 例如:“123adc4".
针对这种情况,我们只要加上判断就行了,只要遇到不是数字字符的直接返回。

2、如果在数字字符前面有正负号又该怎么办???
 例如:”-123”、“+123”.
针对这种情况,我们再加上判断,判断字符串的第一个字符是不是正负号,并用一个标记位flag记录正负。

3、当输入的字符串前面几个字符都是空格又该怎么办???
 例:“   -123”,“   +123”.
  针对这种情况,库里面的atoi是将空格跳过再进行判断的,所以我们在一开始就用循环将空格跳过,注意这时候要是字符串里面全部是空格,则我们应该返回一个值,就返回0吧。

4、当输入的字符串中没有数字时,我们应该返回什么呢???
我们在这种情况下返回0.

5、当字符串中的字符数字转化的整数太大,会有可能溢出???
针对这种情况,我们需要做出一个判断,判断转化后的数字当它的范围在long的范围内。

  那么最重要的一个问题来了,当你第3、4中情况下,返回的是0,如果输入“0”返回的也是0,这两个0怎么区分呢???,貌似在这里你把3、4两种情况下的返回值设置成什么都不合适。所以我们可以设置一个全局变量state,来记录是不是正常返回。

经过以上分析,我们可以写出以下代码:
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
int state=0;                 //用来记录是不是正常返回
long my_atoi(char *arr)
{
	assert(arr);
	long num = 0;             
	int flag = 1;             //设置标记
	while (*arr==' ')              //跳过字符串开头的空格
	{
		arr++;
	}
	if (*arr == '\0')         //如果全部字符串中都是空格
	{
		state = 1;            //将state设置成1,表示异常返回
		return 0;
	}
	if (*arr == '-')          //如果跳过空格,再判断有没有正负
	{
		flag = -1;            //如果是-,则将flag设置成-1
		arr++;
	}
	else if (*arr == '+')
	{
		arr++;
	}
	while (*arr >= '0'&&*arr <= '9'&&*arr)          //判断是不是数字字符
	{
		if ((num>=(signed long)0x80000000) && (num<=(signed long)0x7FFFFFFF)) //判断有没有溢出
		{
			num = num * 10 + flag*(*arr - '0');      //将标记位加上去
			arr++;
		}
		else                                         //如果溢出,则跳出
			break;
	}
	if (*arr != '/0')
	{
		state = 1;                                    //state=1,异常返回
	}
	return num;
}

int main()
{
	char arr[30] = { 0 };
    scanf("%s", arr);
	long ret=my_atoi(arr);
	if (state == 1)
	{
		printf("异常返回\n");
	}
	printf("%ld\n",ret);
	system("pause");
	return 0;
}


  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值