❤️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

相关文章
|
4天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
19 6
|
17天前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。
|
23天前
|
机器学习/深度学习 C语言
【c语言】一篇文章搞懂函数递归
本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
53 7
|
23天前
|
存储 编译器 程序员
【c语言】函数
本文介绍了C语言中函数的基本概念,包括库函数和自定义函数的定义、使用及示例。库函数如`printf`和`scanf`,通过包含相应的头文件即可使用。自定义函数需指定返回类型、函数名、形式参数等。文中还探讨了函数的调用、形参与实参的区别、return语句的用法、函数嵌套调用、链式访问以及static关键字对变量和函数的影响,强调了static如何改变变量的生命周期和作用域,以及函数的可见性。
29 4
|
20天前
|
存储 C语言
【c语言】字符串函数和内存函数
本文介绍了C语言中常用的字符串函数和内存函数,包括`strlen`、`strcpy`、`strcat`、`strcmp`、`strstr`、`strncpy`、`strncat`、`strncmp`、`strtok`、`memcpy`、`memmove`和`memset`等函数的使用方法及模拟实现。文章详细讲解了每个函数的功能、参数、返回值,并提供了具体的代码示例,帮助读者更好地理解和掌握这些函数的应用。
18 0
|
20天前
|
C语言
【c语言】qsort函数及泛型冒泡排序的模拟实现
本文介绍了C语言中的`qsort`函数及其背后的回调函数概念。`qsort`函数用于对任意类型的数据进行排序,其核心在于通过函数指针调用用户自定义的比较函数。文章还详细讲解了如何实现一个泛型冒泡排序,包括比较函数、交换函数和排序函数的编写,并展示了完整的代码示例。最后,通过实际运行验证了排序的正确性,展示了泛型编程的优势。
18 0
|
23天前
|
算法 C语言
factorial函数c语言
C语言中实现阶乘函数提供了直接循环和递归两种思路,各有优劣。循环实现更适用于大规模数值,避免了栈溢出风险;而递归实现则在代码简洁度上占优,但需警惕深度递归带来的潜在问题。在实际开发中,根据具体需求与环境选择合适的实现方式至关重要。
21 0
|
1月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
32 3
|
C语言 机器学习/深度学习 Go
C语言atoi()和itoa()函数的实现
<p>makefile</p> <p></p> <pre code_snippet_id="622931" snippet_file_name="blog_20150319_1_2637659" name="code" class="cpp">.SUFFIXES: .c .o CC=gcc SRCS=test.c OBJS=$(SRCS:.c=.o) EXEC=test all
2272 0
|
24天前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
33 10