atoi()详解及其模拟实现

简介: atoi()详解及其模拟实现

atoi()详解及其模拟实现

atoi()解析

所需头文件为<stdlib.h>

int atoi (const char * str);

整体功能:

分析字符串 str,将其内容解释为整数,该整数作为 int 类型的值返回。

细节:

该函数首先根据需要丢弃尽可能多的空格字符(如在 isspace 中),直到找到第一个非空格字符。

然后,从此字符开始,取一个可选的首字母加号或减号,后跟尽可能多的 10 进制数字,并将它们解释为数值。

字符串可以在构成整数的字符之后包含其他字符,这些字符将被忽略,并且对此函数的行为没有影响。

如果 str 中的第一个非空格字符序列不是有效的整数,或者由于 str 为空或仅包含空格字符而不存在此类序列,则不执行转换并返回零

如果被解释成的数字大于INT_MAX或者小于MIN_MAX,那么就返回0

例如:

#include<stdio.h>
#include<stdlib.h>
int main()
{
    printf("%d\n", atoi("  "));
    printf("%d\n", atoi("0"));
    printf("%d\n", atoi("1234"));
    printf("%d\n", atoi("1234aabcd"));
    printf("%d\n", atoi("  1234  1234"));
    printf("%d\n", atoi("01234"));
    printf("%d\n", atoi("-01234"));
    printf("%d\n", atoi("a1234"));
    printf("%d\n", atoi("a12341111111111111111111111111111111111111111111111111111111"));
    return 0;
}

output:

0
0
1234
1234
1234
1234
-1234
0
0

缺陷:

我们看到,当转换失败时,函数atoi()的返回值是0,正确将字符0转换为数字0,函数atoi()的返回值也是0,因此当函数返回0时,我们就不能区分到底是这个字符串是非法字符串,还是本身转换过来就是0,这点需要在模拟实现中进行改进。

模拟实现my_atoi()

  • 为了避免上面提到的缺陷,我们可以定义一个全局变量state,用来对函数返回值合法性判断:
enum State
{
    VALID,  //0,合法
    INVALID //1,非法
};
enum State state = INVALID; //首先默认非法,如果合法再进行修改
  • 确保传入的指针不是空指针:
assert(str);
  • 排除空格:
while (*str == ' ')
    str++;
  • 如果排除完空格之后,指针str就走到尾了,就说明这是一个空白字符串,直接返回0
if (*str == '\0')
    return 0;
  • 为了方便对正负数进行判断,我们设立一个标记flag
int flag = 1;
if (*str == '-')
{
    flag = -1;
    str++;
}
if (*str == '+')
{
    str++;
}
  • 因为转换数字的时候可能超出int型的范围,因此我们要将返回值定义为long long类型,方便后续进行有效性判断(返回值必须在int型的范围内)
long long ret = 0;
while (isdigit(*str)) //isdigit()用来判断一个字符是否是数字字符
{
    ret = ret * 10 + flag * (*str - '0');
    if (ret > INT_MAX || ret < INT_MIN) //如果结果大于int最大值,小于int最小值,那么直接返回0
        return 0;
    str++;
}
  • 最后,得到返回值ret,再对其后面的字符进行判断,如果是结束符‘\0’,说明返回值合法,否则还是一个非法值
if (*str == '\0')
{
    state = VALID;  //如果合法,就将state修改为VALID,说明返回值合法
    return (int)ret;
}
else
    return (int)ret;

实现代码:

#include<assert.h>
#include<stdlib.h>
#include<stdio.h>
#include<limits.h>
enum State
{
    VALID,//0
    INVALID//1
};
enum State state = INVALID;
int my_atoi(const char* str)
{
    assert(str);
    while (*str == ' ')
        str++;
    if (*str == '\0')
        return 0;
    int flag = 1;
    if (*str == '-')
    {
        flag = -1;
        str++;
    }
    if (*str == '+')
    {
        str++;
    }
    long long ret = 0;
    while (isdigit(*str))
    {
        ret = ret * 10 + flag * (*str - '0');
        if (ret > INT_MAX || ret < INT_MIN)
            return 0;
        str++;
    }
    if (*str == '\0')
    {
        state = VALID;
        return (int)ret;
    }
    else
        return (int)ret;
}
相关文章
|
存储 消息中间件 缓存
键值(key-value)数据库
【4月更文挑战第9天】键值数据库(NoSQL)以键值对形式存储数据,简单灵活,适合任意类型数据。其特点是高性能、高可扩展性,常见应用包括缓存、会话管理、分布式锁、计数统计、配置管理和轻量级消息队列。然而,它不适用于结构化信息存储和复杂查询,选择时需考虑应用场景。
2091 5
|
网络协议 Shell Linux
【Shell 命令集合 网络通讯 】Linux 提供SMB共享 smbd命令 使用指南
【Shell 命令集合 网络通讯 】Linux 提供SMB共享 smbd命令 使用指南
1134 0
|
10月前
|
人工智能 搜索推荐 API
node-DeepResearch:开源复现版OpenAI Deep Research,支持多步推理和复杂查询的AI智能体
node-DeepResearch 是一个开源 AI 智能体项目,支持多步推理和复杂查询,帮助用户逐步解决问题。
1059 27
node-DeepResearch:开源复现版OpenAI Deep Research,支持多步推理和复杂查询的AI智能体
|
存储 编译器 Serverless
C 标准库 - <stdarg.h>详解
`&lt;stdarg.h&gt;` 是 C 标准库中的头文件,提供了处理可变参数函数(varargs)的机制,允许开发者定义接受任意数量参数的函数。它定义了三个主要宏:`va_start`、`va_arg` 和 `va_end`,用于初始化、访问和清理可变参数列表。
|
Linux 开发者 iOS开发
深度剖析:Python如何优雅地跨越操作系统鸿沟,实现无缝对接
【10月更文挑战第2天】Python 作为一种高级编程语言,具备出色的跨平台特性,可在 Windows、macOS 和 Linux 上无缝运行。本文探讨 Python 如何通过核心特性和第三方库实现跨平台开发。内置模块如 `os` 和 `platform` 以及现代库 `pathlib`,简化了文件系统操作。对于 GUI 开发,Tkinter、PyQt 和 wxPython 等库提供了强大支持,确保应用程序的一致性和可移植性。通过具体示例,本文展示了如何利用这些工具应对不同操作系统的需求。
232 6
|
数据采集 Java Python
Python并发编程:多线程(threading模块)
本文详细介绍了Python的threading模块,包括线程的创建、线程同步、线程池的使用,并通过多个示例展示了如何在实际项目中应用这些技术。通过学习这些内容,您应该能够熟练掌握Python中的多线程编程,提高编写并发程序的能力。 多线程编程可以显著提高程序的并发性能,但也带来了新的挑战和问题。在使用多线程时,需要注意避免死锁、限制共享资源的访问,并尽量使用线程池来管理和控制线程。
|
机器学习/深度学习 人工智能 弹性计算
什么是阿里云GPU云服务器?GPU服务器优势、使用和租赁费用整理
阿里云GPU云服务器提供强大的GPU算力,适用于深度学习、科学计算、图形可视化和视频处理等多种场景。作为亚太领先的云服务提供商,阿里云的GPU云服务器具备灵活的资源配置、高安全性和易用性,支持多种计费模式,帮助企业高效应对计算密集型任务。
1816 6
|
JavaScript 前端开发 API
JavaScript基础-事件监听与处理
【6月更文挑战第11天】本文介绍了JavaScript事件驱动编程的核心,包括事件流(捕获、目标、冒泡阶段)和监听方法(DOM Level 0、addEventListener/removeEventListener)。讨论了常见问题和易错点,如内存泄漏、事件委托和阻止默认行为的混淆,并提供了解决策略。通过代码示例展示了事件绑定、事件委托和阻止默认行为的用法,强调理解事件处理机制对于编写高效交互式Web应用的重要性。
899 6
|
存储 算法 图形学
【计算机图形学】实验二 用扫描线算法实现多边形填充
【计算机图形学】实验二 用扫描线算法实现多边形填充
951 2
|
传感器 芯片
嵌入式通信协议全解析:SPI、I²C、UART详解(附带面试题)
通信是指人与人或人与自然之间通过某种行为或媒介进行的信息交流与传递。从广义上来说,通信是指需要信息的双方或多方在不违背各自意愿的情况下采用任意方法、任意媒质,将信息从某方准确安全地传送到另方。在出现电波传递通信后,通信被单一解释为信息的传递,是指由一地向另一地进行信息的传输与交换,其目的是传输消息。通信方式包括利用“电”来传递消息的电信,这种通信具有迅速、准确、可靠等特点,且几乎不受时间、地点、空间、距离的限制,因而得到了飞速发展和广泛应用。
4842 0