LeetCode算法题:8.字符串转换整数 (atoi)

2024-05-08 02:52

本文主要是介绍LeetCode算法题:8.字符串转换整数 (atoi),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

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

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

注意:

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

示例 1:

输入:s = "42"
输出:42
解释:加粗的字符串为已经读入的字符,插入符号是当前读取的字符。
第 1 步:"42"(当前没有读入字符,因为没有前导空格)^
第 2 步:"42"(当前没有读入字符,因为这里不存在 '-' 或者 '+')^
第 3 步:"42"(读入 "42")^
解析得到整数 42 。
由于 "42" 在范围 [-2^31, 2^31 - 1] 内,最终结果为 42 。

示例 2:

输入:s = "   -42"
输出:-42
解释:
第 1 步:"   -42"(读入前导空格,但忽视掉)^
第 2 步:"   -42"(读入 '-' 字符,所以结果应该是负数)^
第 3 步:"   -42"(读入 "42")^
解析得到整数 -42 。
由于 "-42" 在范围 [-2^31, 2^31 - 1] 内,最终结果为 -42 。

示例 3:

输入:s = "4193 with words"
输出:4193
解释:
第 1 步:"4193 with words"(当前没有读入字符,因为没有前导空格)^
第 2 步:"4193 with words"(当前没有读入字符,因为这里不存在 '-' 或者 '+')^
第 3 步:"4193 with words"(读入 "4193";由于下一个字符不是一个数字,所以读入停止)^
解析得到整数 4193 。
由于 "4193" 在范围 [-2^31, 2^31 - 1] 内,最终结果为 4193 。

提示:

  • 0 <= s.length <= 200
  • s 由英文字母(大写和小写)、数字(0-9)、' ''+''-''.' 组成

我的解题思路:
正常按逻辑处理就行,先跳过前面的空格和‘0’,然后检查是否有‘-’,没有的话则按正数处理,有的话则最后返回时添加去相反数。然后逐个遍历计算相加,直到不是数字停止。检查数值是否越界,做越界处理。最后返回值

代码:

class Solution {public int myAtoi(String s) {char[] cs = s.toCharArray();int len = s.length();int i;boolean isNegative = false;long result = 0;for ( i = 0; i < len && cs[i]==' '; i++) {// 跳过空格}if(cs[i]=='-'){// 有负号isNegative = true;i++;}while (i<len){if(cs[i] >= '0'&& cs[i]<='9'){// 是数字result = result*10 + (cs[i]-'0');//}else{// 不是数字结束break;}i++;}if(result<Integer.MIN_VALUE){result = Integer.MIN_VALUE;}if (isNegative) result = -result;return (int)result;}
}

出现错误:
在这里插入图片描述
错误原因:最后判断越界的逻辑写错了,应该是判断是否大于int的最大值,因为之前都没有进行负数处理,所以result一直是正数。

class Solution {public int myAtoi(String s) {char[] cs = s.toCharArray();int len = s.length();int i;boolean isNegative = false;long result = 0;for ( i = 0; i < len && cs[i]==' '; i++) {// 跳过空格}if(cs[i]=='-'){// 有负号isNegative = true;i++;}while (i<len){if(cs[i] >= '0'&& cs[i]<='9'){// 是数字result = result*10 + (cs[i]-'0');//}else{// 不是数字结束break;}i++;}if (isNegative) result = -result;if(result>=Integer.MAX_VALUE){result = Integer.MAX_VALUE;}if (result<=Integer.MIN_VALUE){result = Integer.MIN_VALUE;}return (int)result;}
}

再次出现错误:
在这里插入图片描述
错误原因:当字符串为空时造成数组下标越界。因此需要在一开始对空数组直接返回0处理。
再次出现错误:
在这里插入图片描述
错误原因,之前修改只是简单的对字符串判空,但是存在跳过空格之后为空字符的情况,因此,可以将判空放在跳过空字符之后。
继续出错:
在这里插入图片描述
错误原因:数值太大,造成long类型都越界且变成负数,因此上述判断是否大于int类型最大值的逻辑不太合理。

整理完上述错误逻辑之后重新梳理解题逻辑:
先将前面的空格全部跳过,然后判断字符串是否遍历完,遍历完则直接返回0,没遍历完则判断下一个字符是否是’+'或者‘-’,然后再对剩余字符遍历。停止条件有三个:

  1. 数值越界,即负数小于-231,或者正数大于231-1
  2. 遇到非数字字符。
  3. 正常遍历完了结束。

对正数和负数分别处理。最终正确题解:

class Solution {public int myAtoi(String s) {char[] cs = s.toCharArray();int len = s.length();int i;boolean isNegative = false;int result = 0;for ( i = 0; i < len && cs[i]==' '; i++) {// 跳过空格}if (i==len){// 字符串已经遍历完了return 0;}if (cs[i]=='-'||cs[i]=='+'){// 有正负号if (cs[i]=='-')isNegative = true;i++;}if(isNegative){// 负数处理逻辑while (i<len){if (cs[i]>='0'&& cs[i]<='9'){//是数字if (result<Integer.MIN_VALUE/10){// 小于最小值了,直接返回最小值result = Integer.MIN_VALUE;return result;}result = result*10 - (cs[i]-'0');if(result>0 ){// 小于最小值了,直接返回最小值result = Integer.MIN_VALUE;return result;}}else{// 不是数字,直接结束遍历break;}i++;}}else{// 正数处理逻辑while (i<len){if (cs[i]>='0'&& cs[i]<='9'){//是数字if (result>Integer.MAX_VALUE/10){result = Integer.MAX_VALUE;return result;}result = result*10 + (cs[i]-'0');if(result<0 ){// 大于最小值了,直接返回最大值result = Integer.MAX_VALUE;return result;}}else{// 不是数字,直接结束遍历break;}i++;}}// 正常返回return result;}
}

在这里插入图片描述
自己题解总结:
1.对于数值越界处理,对正数和负数分开处理,正数时计算之前判断是否大于最大值/10,计算之后判断是否为负数发生越界。负数时计算之前判断是否小于最小值/10,计算之后判断是否为正数发生越界。
2.对于前一步有两个逻辑判断的步骤,下一步必须得考虑清楚每一个逻辑,不落下没处理的逻辑。例如在这里插入图片描述
这里跳过空格有两个逻辑判断都可能结束遍历,则下一步得区分是哪一个条件导致的遍历结束。

官方题解:

class Solution {public int myAtoi(String str) {Automaton automaton = new Automaton();int length = str.length();for (int i = 0; i < length; ++i) {automaton.get(str.charAt(i));}return (int) (automaton.sign * automaton.ans);}
}class Automaton {public int sign = 1;public long ans = 0;private String state = "start";private Map<String, String[]> table = new HashMap<String, String[]>() {{put("start", new String[]{"start", "signed", "in_number", "end"});put("signed", new String[]{"end", "end", "in_number", "end"});put("in_number", new String[]{"end", "end", "in_number", "end"});put("end", new String[]{"end", "end", "end", "end"});}};public void get(char c) {state = table.get(state)[get_col(c)];if ("in_number".equals(state)) {ans = ans * 10 + c - '0';ans = sign == 1 ? Math.min(ans, (long) Integer.MAX_VALUE) : Math.min(ans, -(long) Integer.MIN_VALUE);} else if ("signed".equals(state)) {sign = c == '+' ? 1 : -1;}}private int get_col(char c) {if (c == ' ') {return 0;}if (c == '+' || c == '-') {return 1;}if (Character.isDigit(c)) {return 2;}return 3;}
}作者:力扣官方题解
链接:https://leetcode.cn/problems/string-to-integer-atoi/solutions/183164/zi-fu-chuan-zhuan-huan-zheng-shu-atoi-by-leetcode-/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

思路

字符串处理的题目往往涉及复杂的流程以及条件情况,如果直接上手写程序,一不小心就会写出极其臃肿的代码。

因此,为了有条理地分析每个输入字符的处理方法,我们可以使用自动机这个概念:

我们的程序在每个时刻有一个状态 s,每次从序列中输入一个字符 c,并根据字符 c 转移到下一个状态 s’。这样,我们只需要建立一个覆盖所有情况的从 s 与 c 映射到 s’ 的表格即可解决题目中的问题。
作者:力扣官方题解
链接:https://leetcode.cn/problems/string-to-integer-atoi/solutions/183164/zi-fu-chuan-zhuan-huan-zheng-shu-atoi-by-leetcode-/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
在这里插入图片描述
说实话,一开始拿到这一题想到过可能可以用这方面去写,但是关于原理完全忘了,也就不敢去写,结果自己一个一个逻辑判断却总是存在漏洞,官方题解太优雅了,对于大多数人应该也都是看一看。这一题对于我更多的启发是在于如何判断int数据溢出。
官方使用了,其中ans为long类型,这就不会造成ans在计算时溢出,然后去区分正负数的情况,正数与int的最大值比较取小,负数与int的最小值的相反数比较取小

ans = sign == 1 ? Math.min(ans, (long) Integer.MAX_VALUE) : Math.min(ans, -(long) Integer.MIN_VALUE);

如果限制了为int类型,则可以使用我上述的方式计算前判断是否将会溢出,以及计算后的值是否溢出。

这篇关于LeetCode算法题:8.字符串转换整数 (atoi)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/969106

相关文章

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

C#中字符串分割的多种方式

《C#中字符串分割的多种方式》在C#编程语言中,字符串处理是日常开发中不可或缺的一部分,字符串分割是处理文本数据时常用的操作,它允许我们将一个长字符串分解成多个子字符串,本文给大家介绍了C#中字符串分... 目录1. 使用 string.Split2. 使用正则表达式 (Regex.Split)3. 使用

Java将时间戳转换为Date对象的方法小结

《Java将时间戳转换为Date对象的方法小结》在Java编程中,处理日期和时间是一个常见需求,特别是在处理网络通信或者数据库操作时,本文主要为大家整理了Java中将时间戳转换为Date对象的方法... 目录1. 理解时间戳2. Date 类的构造函数3. 转换示例4. 处理可能的异常5. 考虑时区问题6.

Java中JSON字符串反序列化(动态泛型)

《Java中JSON字符串反序列化(动态泛型)》文章讨论了在定时任务中使用反射调用目标对象时处理动态参数的问题,通过将方法参数存储为JSON字符串并进行反序列化,可以实现动态调用,然而,这种方式容易导... 需求:定时任务扫描,反射调用目标对象,但是,方法的传参不是固定的。方案一:将方法参数存成jsON字

基于C#实现将图片转换为PDF文档

《基于C#实现将图片转换为PDF文档》将图片(JPG、PNG)转换为PDF文件可以帮助我们更好地保存和分享图片,所以本文将介绍如何使用C#将JPG/PNG图片转换为PDF文档,需要的可以参考下... 目录介绍C# 将单张图片转换为PDF文档C# 将多张图片转换到一个PDF文档介绍将图片(JPG、PNG)转

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖