草庐IT

模拟实现atoi函数(将数字字符串转换为整型)附加leedcode练习题

不能再留遗憾了 2023-06-15 原文

各位朋友们,大家好啊!今天我为大家分享的知识是如何模拟实现atoi函数。相信大家如果能够理解这个知识,对大家以后的刷题是有帮助的。

文章目录

什么是atoi函数(atoi函数的作用)

我们要想实现某个函数,我们肯定要先知道这个函数的作用是什么,然后我们再根据它的作用来自己实现。我们先来看看stoi函数在库函数中是怎么样的吧。

int atoi (const char * str);

这函数的参数只有一个字符串的地址,它的返回类型是一个整型类型。

这句话的意思是:
该函数首先根据需要丢弃尽可能多的空格字符(如在 isspace 中),直到找到第一个非空格字符。然后,从此字符开始,取一个可选的首字母加号或减号,后跟尽可能多的 10 进制数字,并将它们解释为数值。

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

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

先直接使用库函数看看这个函数是什么作用

我们要想使用atoi这个函数,我们需要引用#include这个头文件。

都是正整数字符的字符串

#include<stdio.h>
#include<stdlib.h>

int main()
{
	char* arr = "1234";   //我们先举一个正整数的例子
	int ret = atoi(arr);
	printf("%d\n", ret);

	return 0;
}

含有负号的整数字符的字符串

#include<stdio.h>
#include<stdlib.h>

int main()
{
	char* arr = "-1234";   //负整数
	int ret = atoi(arr);
	printf("%d\n", ret);

	return 0;
}

含有非数字字符的字符串,且非数字字符都在一起

#include<stdio.h>
#include<stdlib.h>

int main()
{
	char* arr = "-1234#abd";
	int ret = atoi(arr);
	printf("%d\n", ret);

	return 0;
}

数字字符跟非数字字符交叉出现

#include<stdio.h>
#include<stdlib.h>

int main()
{
	char* arr = "-1234a23cd";  //数字跟字母交叉出现
	int ret = atoi(arr);
	printf("%d\n", ret);

	return 0;
}


通过这几种代码,我们大概可以知道,atoi函数在遇到非数字字符就会停止读取了。

模拟实现atoi函数

我们实现atoi函数主要针对这些问题:

  1. 空指针。当传入的参数是NULL时,我们应该对其做出判断,以防出现非法访问。
  2. 空字符串。当传入的字符串为空时,我们就返回0
      • .当传入的字符串中第一个字符时’+‘或者’-'的时候,我们需要判断这个字符串的正负性。
  3. 非数字字符。当字符串中出现了非数字字符的时候,我们需要直接停止当前函数并返回已经转换完成的数字。
  4. 越界。因为在库函数中,atoi函数返回的是int类型,但是当我们传入的字符串长度很长时,那么这个字符串在转换为整形的时候,可能会出现超过int所表示的范围的时候。这时候我们就可以停止当前函数并返回已经转换完成的整数。下面就是代码实现。
#include<stdio.h>
#include<limits.h>  //int类型的取值范围
#include<ctype.h>   //判断是否为数字字符
#include<assert.h>  //判断传入的字符串是否为NULL

//我们定义一个枚举来判断最终返回的结果是否是正常读取结束的
enum S
{
	VALID,
	INVALID
};

enum S s = INVALID;

int my_atoi(char* str)
{
	assert(str);
	if (*str == '\0')
	{
		return 0;
	}
	//这个flag是来决定你最终的整数的正负性
	int flag = 1;
	if (*str == '-')
	{
		flag = -1;
		str++;
	}
	else if (*str == '+')
	{
		flag = 1;
		str++;
	}
	long long ret = 0;
	while (*str != '\0')
	{
		if (isdigit(*str))
		{
			ret = ret * 10 + flag * (*str - '0');
			if (ret > INT_MAX || ret < INT_MIN)
			{
				return (int)ret;
			}
		}
		else
		{
			return (int)ret;
		}
		str++;
	}
	if (*str == '\0')
	{
		s = VALID;
	}
	return (int)ret;
}

int main()
{
	char* arr = "-1234a5s3gd";
	int ret = my_atoi(arr);
	if (s == INVALID)
	{
		printf("坐标非法:%d\n", ret);
	}
	else
	{
		printf("合法转换:%d\n", ret);
	}

	return 0;
}

leedcode相关题目

leedcode之字符串转换整数 (atoi)

题目要求

请你来实现一个 myAtoi(string s) 函数,使其能将字符串转换成一个 32 位有符号整数(类似 C/C++ 中的 atoi 函数)。

函数 myAtoi(string s) 的算法如下:

读入字符串并丢弃无用的前导空格
检查下一个字符(假设还未到字符末尾)为正还是负号,读取该字符(如果有)。 确定最终结果是负数还是正数。 如果两者都不存在,则假定结果为正。
读入下一个字符,直到到达下一个非数字字符或到达输入的结尾。字符串的其余部分将被忽略。
将前面步骤读入的这些数字转换为整数(即,“123” -> 123, “0032” -> 32)。如果没有读入数字,则整数为 0 。必要时更改符号(从步骤 2 开始)。
如果整数数超过 32 位有符号整数范围 [−231, 231 − 1] ,需要截断这个整数,使其保持在这个范围内。具体来说,小于 −231 的整数应该被固定为 −231 ,大于 231 − 1 的整数应该被固定为 231 − 1 。
返回整数作为最终结果。

注意:

本题中的空白字符只包括空格字符 ’ ’ 。
除前导空格或数字后的其余字符串外,请勿忽略 任何其他字符。

代码实现

int myAtoi(char * s){
	assert(s);
	//判断是否为空字符串
	if (*s == '\0')
	{
		return 0;
	}
	//因为他给的字符串可能开始就是空格,所以我们需要跳过这些空格
	while(*s == ' ')
	{
		s++;
	}
	int flag = 1;
	if (*s == '-')
	{
		flag = -1;
		s++;
	}
	else if (*s == '+')
	{
		flag = 1;
		s++;
	}
	long long ret = 0;
	while (*s != '\0')
	{
		if (isdigit(*s))
		{
			ret = ret * 10 + flag * (*s - '0');
			if (ret >= INT_MAX)
			{
	//这里注意题目的要求,如果超过int的范围,如果大于2^31-1,就返回2^31-1
				return INT_MAX;
			}
			else if(ret <= INT_MIN)
			{
			//小于-2^31就返回-2^31
				return INT_MIN;
			}
		}
		else
		{
			return (int)ret;
		}
		s++;
	}
	return (int)ret;
}

小结

大家可以自己去写写这道题,巩固下这个知识。那么到这里,我的这次分享就结束了,如果有错误,欢迎大家指出来,如果觉得博主写的不错记得给个赞哦。非常感谢!!!

有关模拟实现atoi函数(将数字字符串转换为整型)附加leedcode练习题的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. Ruby 解析字符串 - 2

    我有一个字符串input="maybe(thisis|thatwas)some((nice|ugly)(day|night)|(strange(weather|time)))"Ruby中解析该字符串的最佳方法是什么?我的意思是脚本应该能够像这样构建句子:maybethisissomeuglynightmaybethatwassomenicenightmaybethiswassomestrangetime等等,你明白了......我应该一个字符一个字符地读取字符串并构建一个带有堆栈的状态机来存储括号值以供以后计算,还是有更好的方法?也许为此目的准备了一个开箱即用的库?

  3. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

    我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

  4. ruby-on-rails - unicode 字符串的长度 - 2

    在我的Rails(2.3,Ruby1.8.7)应用程序中,我需要将字符串截断到一定长度。该字符串是unicode,在控制台中运行测试时,例如'א'.length,我意识到返回了双倍长度。我想要一个与编码无关的长度,以便对unicode字符串或latin1编码字符串进行相同的截断。我已经了解了Ruby的大部分unicode资料,但仍然有些一头雾水。应该如何解决这个问题? 最佳答案 Rails有一个返回多字节字符的mb_chars方法。试试unicode_string.mb_chars.slice(0,50)

  5. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  6. ruby-on-rails - Rails 常用字符串(用于通知和错误信息等) - 2

    大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje

  7. ruby - 如何以所有可能的方式将字符串拆分为长度最多为 3 的连续子字符串? - 2

    我试图获取一个长度在1到10之间的字符串,并输出将字符串分解为大小为1、2或3的连续子字符串的所有可能方式。例如:输入:123456将整数分割成单个字符,然后继续查找组合。该代码将返回以下所有数组。[1,2,3,4,5,6][12,3,4,5,6][1,23,4,5,6][1,2,34,5,6][1,2,3,45,6][1,2,3,4,56][12,34,5,6][12,3,45,6][12,3,4,56][1,23,45,6][1,2,34,56][1,23,4,56][12,34,56][123,4,5,6][1,234,5,6][1,2,345,6][1,2,3,456][123

  8. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  9. ruby - 如何使用文字标量样式在 YAML 中转储字符串? - 2

    我有一大串格式化数据(例如JSON),我想使用Psychinruby​​同时保留格式转储到YAML。基本上,我希望JSON使用literalstyle出现在YAML中:---json:|{"page":1,"results":["item","another"],"total_pages":0}但是,当我使用YAML.dump时,它不使用文字样式。我得到这样的东西:---json:!"{\n\"page\":1,\n\"results\":[\n\"item\",\"another\"\n],\n\"total_pages\":0\n}\n"我如何告诉Psych以想要的样式转储标量?解

  10. ruby 正则表达式 - 如何替换字符串中匹配项的第 n 个实例 - 2

    在我的应用程序中,我需要能够找到所有数字子字符串,然后扫描每个子字符串,找到第一个匹配范围(例如5到15之间)的子字符串,并将该实例替换为另一个字符串“X”。我的测试字符串s="1foo100bar10gee1"我的初始模式是1个或多个数字的任何字符串,例如,re=Regexp.new(/\d+/)matches=s.scan(re)给出["1","100","10","1"]如果我想用“X”替换第N个匹配项,并且只替换第N个匹配项,我该怎么做?例如,如果我想替换第三个匹配项“10”(匹配项[2]),我不能只说s[matches[2]]="X"因为它做了两次替换“1fooX0barXg

随机推荐