❤️C语言模拟实现atoi函数❤️

简介: 大家国庆节快乐!本篇博客由博主来为大家讲解关于atoi函数,不知道大家对这个函数了解吗?我根据大佬们写的atoi的模拟实现,整理出其中的相关知识。

💐atoi的用法



atoi函数原型为:int atoi( constchar *string );


其功能是将一个数字字符串转换成int类型的整数,若数字前有空格,可以跳过空格。


💐具体分析:


首先,该函数会根据需要丢弃无用的开头空格字符,直到寻找到第一个非空格的字符为止。

接下来的转化规则如下:

如果第一个非空字符为正或者负号时,则将该符号与之后面尽可能多的连续数字字符组合起来,形成一个有符号整数。

假如第一个非空字符是数字,则直接将其与之后连续的数字字符组合起来,形成—个整数。

该字符串在有效的整数部分之后也可能会存在多余的字符,那么这些字符可以被忽略,它们对函数不应该造成影响。

注意:假如该字符串中的第一个非空格字符不是一个有效整数字符、字符串为空或字符串仅包含空白字符时,则你的函数不需要进行转换,即无法进行有效转换。

在任何情况下,若函数不能进行有效的转换时,请返回0。


💐一些相关知识


🌺🌺1.assert断言


assert宏的原型定义在<assert.h>中,其作用是如果它的条件返回错误,则终止程序执行,并返回错误原因。


原型定义:

#include <assert.h>
void assert( int expression );

本题模拟的方法一中就含有assert对空指针的断言。

这就体现了使用assert的好处:


空指针检查。例如,针对一个函数的参数进行空指针检查。你可以这样使用:assert (pointer != NULL);,产生的错误会像这样:Assertion ‘pointer != ((void *)0)' failed

这样,当出现空指针时,你的程序就会退出,并很好的给出错误信息。


检查函数参数的值。例如,如果一个函数只能在它的一个参数foo为正值的时候被调用,你可以在函数开始时这样写:assert (foo > 0);,这将帮助你检测函数的错误使用,这也给源代码阅读者很清晰的印象,那就是在这里对函数的参数值有限制。


🌺🌺2.isspace函数


💐原型:


int isspace ( int c );

💐用法:

#include  <ctype.h>

💐功能:

判断字符c是否为空白符

💐说明:

  1. 当c为空白符时,返回非零值,否则返回零。(空白符指空格、水平制表、垂直制表、换页、回车和换行符。)
  2. 需要引用头文件 <ctype.h>


🌺🌺3.isdigit函数


💐原型:

int isdigit ( int c );

💐用法:

#include  <ctype.h>

💐功能:

检测输入的字符串是否只由数字组成。


💐说明:


当c是介于0~9的数字时,返回一个非零值,否则返回零。

需要引用头文件 <ctype.h>


🌺🌺4.enum枚举类型


枚举:就是一一列举。

枚举常量:{ }中的内容是枚举类型的可能取值,就叫枚举常量 。

枚举常量都是有值的,默认从0开始,一次递增1,当然在定义的时候也可以赋初值。


如果你还想深究的话----请点击这篇博客哦----


🌺🌺5.INT_MAX和INT_MIN


C/C++中常量INT_MAX和INT_MIN已经定义了,分别表示最大、最小整数.


💐头文件:

#include limits.h

💐代表的值为:

INT_MAX=2^32-1=2147483647
INT_MIN=-2^32=-2147483648

💐两者在头文件limit.h中定义为:

#define INT_MAX 2147483647
#define INT_MIN (-INT_MAX-1)

在C/C++语言中,不能够直接使用-2147483648来代替最小负数,因为这不是一个数字,而是一个表达式。表达式的意思是对整数21473648取负,但是2147483648已经溢出了int的上限,所以定义为(-INT_MAX -1)。


故引出溢出的情况:

C中int类型是32位的,范围是-2147483648 ~ 2147483647 。


(1)最轻微的上溢是INT_MAX + 1 :结果是 INT_MIN;

(2)最严重的上溢是INT_MAX + INT_MAX :结果是-2;

(3)最轻微的下溢是INT_MIN - 1:结果是INT_MAX;

(4)最严重的下溢是INT_MIN +INT_MIN:结果是0 。


可见他们构成一个整数环,且存在方向。


💐实现atoi需要注意的点




🌺🌺1.考虑传参时传了空指针


💐解决方法:

使用assert进行断言操作。

  //空指针
  assert(str != NULL);

需要引用头文件:#include <assert.h>


🌺🌺2.考虑负数


💐解决方法:

引入新变量flag,当判断有负号的时候,令flag=-flag

  //+-号
  if (*str == '+')
    str++;
  if (*str == '-')
  {
    flag = -flag;
    str++;
  }


🌺🌺3.考虑空字符串


💐解决方法:

使用库函数

  if (*str == '\0')
    return 0;   //返回结果为非法


🌺🌺4.考虑字符串前有空格


💐解决方法:

  while (isspace(*str))
  {
    str++;
    if (*str == '\0')
      return 0;//字符串只有空格
  }


🌺🌺5.考虑有其他非数字字符


💐解决方法:

如果有非数字和非空格的字符,那就只返回该字符前面的符号。

else//非数字字符
  {
    state = VALID;
    return (int)ret;//返回结果为前面的数字
  }


💐aoti测试的实例



🌺🌺1.输入“42”

38beb22db60f42339a030459ca491a31.png


🌺🌺2.输入“ -42”


1e4e170310f547a5905cfd7aee5f85b0.png


🌺🌺3.输入“4193 with words”


5a5fbcba1ff044bcb2a342b8862fc877.png


🌺🌺4.输入“-91283472332”


24a10142d98147d787693e61bf2a20a5.png


💐atoi的模拟实现



#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <ctype.h>
enum STATE
{
  VALID,//合法
  INVALID//非法
};
enum STATE state = INVALID;
int my_atoi( char* str)
{
  int flag = 1;
  long long ret = 0;
  //空指针
  assert(str != NULL);
  //处理空字符串
  if (*str == '\0')
    return 0;//返回结果为非法
  //处理空格
  while (isspace(*str))
  {
    str++;
    if (*str == '\0')
      return 0;//字符串只有空格
  }
  //+-号
  if (*str == '+')
    str++;
  if (*str == '-')
  {
    flag = -flag;
    str++;
  }
  //字符串处理
  while (*str)
  {
    if (isdigit(*str))
    {
      ret = ret * 10 + (*str - '0') * flag;
      //溢出
      if ((ret > INT_MAX) || (ret < INT_MIN))
      {
        state = VALID;
        return INT_MIN;
      }
    }
    else//非数字字符
    {
      state = VALID;
      return (int)ret;//返回结果为前面的数字
    }
    str++;
  }
  state = VALID;
  return (int)ret;
}
int main()
{
  char p[] = "4139 with words";
  int ret = my_atoi(p);
  if (state == VALID)//合法时,输出结果
    printf("%d\n", ret);
}

定义全局变量stste,并设置初值为INVALID,在函数实现中当遇到合法的返回时时,将其改为VALID,在测试时加入if判断,这样可以将合法的返回值与非法的返回值区分开。定义ret为long long类型,不能为int 类型,否则,在计算时会发生截断,这样ret永远不会到达int 类型的最大值或最小值。


💐结果

🌺🌺结果一

dc55b60dcd5948db8e3f37c05feee41c.png

🌺🌺结果二74b02b45f727432385a64549949b1bf0.png

🌺🌺结果三

d1f1d736d7314dab8639db0013c57391.png

🌺🌺结果四

7a37a4e68e684cf98597afde4c361733.png

相关文章
|
17天前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
39 10
|
17天前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
41 9
|
17天前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
31 8
|
17天前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
40 6
|
17天前
|
存储 C语言
【C语言】输入/输出函数详解
在C语言中,输入/输出操作是通过标准库函数来实现的。这些函数分为两类:标准输入输出函数和文件输入输出函数。
101 6
|
17天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
47 6
|
17天前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
27 5
|
机器学习/深度学习 人工智能 C语言
C语言简单实现14个例题(谭浩强第四版)
版权声明:转载请注明出处:http://blog.csdn.net/dajitui2024 https://blog.csdn.net/dajitui2024/article/details/79396241 1、仅供学习交流参考。
1165 0
|
2月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
39 3
|
28天前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
50 4