【经典算法】LeetCode 8. 字符串转换整数 (atoi)(Java/C/Python3/Go实现含注释说明,Easy)

news2024/11/26 22:28:47
  • 作者主页: 🔗进朱者赤的博客

  • 精选专栏:🔗经典算法

  • 作者简介:阿里非典型程序员一枚 ,记录在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名

  • ❤️觉得文章还不错的话欢迎大家点赞👍➕收藏⭐️➕评论,💬支持博主,记得点个大大的关注,持续更新🤞
    ————————————————-

目录

  • 题目描述
  • 思路及实现
    • 方式一:状态机
      • 思路
      • 代码实现
        • Java版本
        • C语言版本
        • Python3版本
        • Golang版本
      • 复杂度分析
    • 方式二:正则表达式
      • 思路
      • 代码实现
        • Java
        • C++
        • Python3
        • Go
      • 复杂度分析
  • 总结
  • 相似题目

  • 标签(题目类型):字符串处理、数学

题目描述

将字符串转换成一个整数(符号位为 +-,数位为 0-9),如果字符串不符合整数格式,则返回 0。

原题:LeetCode 8

思路及实现

方式一:状态机

思路

使用状态机来处理字符串的转换,定义几个状态:

  • START:初始状态,等待数字或者符号
  • SIGNED:已读取符号,等待数字
  • IN_NUMBER:已读取数字,继续读取数字
  • END:结束状态

根据当前字符和状态来更新状态,并决定是否更新结果。
在这里插入图片描述

代码实现

Java版本
public class Solution {
    public int myAtoi(String s) {
        // 状态定义
        final int START = 0, SIGNED = 1, IN_NUMBER = 2, END = 3;
        int state = START;
        int index = 0;
        int sign = 1;
        long result = 0;

        while (index < s.length()) {
            char c = s.charAt(index);
            if (state == START) {
                if (c == ' ') {
                    // 忽略空格
                } else if (c == '-' || c == '+') {
                    sign = (c == '-') ? -1 : 1;
                    state = SIGNED;
                } else if (Character.isDigit(c)) {
                    state = IN_NUMBER;
                    result = c - '0';
                } else {
                    // 不合法字符,返回0
                    return 0;
                }
            } else if (state == SIGNED || state == IN_NUMBER) {
                if (Character.isDigit(c)) {
                    // 防止溢出
                    if (result > Integer.MAX_VALUE / 10 || (result == Integer.MAX_VALUE / 10 && c - '0' > 7)) {
                        return (sign == 1) ? Integer.MAX_VALUE : Integer.MIN_VALUE;
                    }
                    result = result * 10 + (c - '0');
                } else {
                    // 非数字字符,结束转换
                    state = END;
                }
            }
            index++;
        }

        return (int) (sign * result);
    }
}

说明:使用状态机来处理字符串的转换,对不同的字符和状态有清晰的判断逻辑。

C语言版本
#include <ctype.h>
#include <limits.h>

int myAtoi(char *s) {
    // 状态定义
    // ...(与Java版本类似)
    // ...
    // 实现逻辑(与Java版本类似)
    // ...
    return sign * (int)(result > INT_MAX ? INT_MAX : result < INT_MIN ? INT_MIN : result);
}

说明:C语言实现与Java类似,但需要手动处理整数溢出。

Python3版本
class Solution:
    def myAtoi(self, s: str) -> int:
        # 状态定义
        # ...(与Java版本类似)
        # ...
        # 实现逻辑(与Java版本类似)
        # ...
        return sign * max(min(result, 2**31 - 1), -2**31)

说明:Python版本也使用状态机,但不需要手动处理整数溢出,因为Python的整数类型可以自动处理大数。

Golang版本
func myAtoi(s string) int {
    // 状态定义
    // ...(与Java版本类似)
    // ...
    // 实现逻辑(与Java版本类似)
    // ...
    if result > math.MaxInt32 {
        return math.MaxInt32
    }
    if result

< math.MinInt32 {
        return math.MinInt32
    }
    return int(sign * result)
}

说明:在Go语言中,我们需要手动处理整数的溢出,因为Go的int类型是有固定大小的(32位或64位,取决于平台)。我们使用math.MaxInt32math.MinInt32来表示32位整数的最大和最小值。

复杂度分析

  • 时间复杂度:O(n),其中n是字符串s的长度。我们需要遍历整个字符串一次。
  • 空间复杂度:O(1),我们只使用了常数个变量来存储状态、符号、结果等,没有使用与输入规模成比例的额外空间。

方式二:正则表达式

思路

使用正则表达式来匹配和提取字符串中的整数部分,然后将其转换为整数。

代码实现

由于方式二提到的使用正则表达式来实现字符串转整数的方法在Python中比较简洁,但您希望看到其他语言的实现,我将提供Java、C++、Go和JavaScript的四种实现方式,并附带注释和说明。

Java
import java.util.regex.Matcher;
import java.util.regex.Pattern;

public class Solution {
    public int myAtoi(String s) {
        // 去除前导空格
        s = s.trim();
        // 定义正则表达式,匹配整数部分(包括正负号)
        Pattern pattern = Pattern.compile("^[\\+\\-]?\\d+");
        Matcher matcher = pattern.matcher(s);
        
        if (!matcher.find()) {
            // 没有找到匹配项,返回0
            return 0;
        }
        
        // 提取匹配的整数部分并转换为长整型
        long num = Long.parseLong(matcher.group());
        
        // 处理溢出情况
        num = Math.max(Math.min(num, Integer.MAX_VALUE), Integer.MIN_VALUE);
        
        // 转换为int类型并返回
        return (int) num;
    }
}
C++
#include <iostream>
#include <string>
#include <regex>
#include <climits>

class Solution {
public:
    int myAtoi(std::string s) {
        // 去除前导空格
        s.erase(0, s.find_first_not_of(" "));
        
        // 定义正则表达式,匹配整数部分(包括正负号)
        std::regex pattern("^[\\+\\-]?\\d+");
        std::smatch match;
        
        if (!std::regex_search(s, match, pattern)) {
            // 没有找到匹配项,返回0
            return 0;
        }
        
        // 提取匹配的整数部分并转换为长整型
        std::string numStr = match.str();
        long long num = std::stoll(numStr);
        
        // 处理溢出情况
        num = std::max(std::min(num, INT_MAX), INT_MIN);
        
        // 转换为int类型并返回
        return static_cast<int>(num);
    }
};
Python3

在Python中,实现一个将字符串转换为整数的函数(类似于int()函数,但更健壮,可以处理异常和边界情况),我们可以编写一个函数,该函数首先去除字符串前后的空格,然后使用正则表达式匹配可能的整数部分(包括正负号),并处理可能的溢出情况。以下是Python的实现:

import re

def my_atoi(s: str) -> int:
    # 去除前导空格
    s = s.strip()
    
    # 定义正则表达式,匹配整数部分(包括正负号)
    pattern = re.compile(r'^[\+\-]?\d+')
    match = pattern.match(s)
    
    if not match:
        # 没有找到匹配项,返回0
        return 0
    
    # 提取匹配的整数部分并转换为int
    num_str = match.group()
    
    # 尝试转换为整数,并处理可能的溢出
    try:
        num = int(num_str)
        # Python的int类型在32位系统上通常是32位,但确保不溢出
        num = max(min(num, 2**31 - 1), -2**31)
    except ValueError:
        # 如果字符串包含非数字字符(尽管这在上面的正则表达式中应该已经排除了),则返回0或默认值
        return 0
    
    # 返回转换后的整数
    return num

# 示例调用
print(my_atoi("   -42"))  # 输出: -42
print(my_atoi("4193 with words"))  # 输出: 4193
print(my_atoi("words and 987"))  # 输出: 0
print(my_atoi("-91283472332"))  # 输出: -2147483648,因为Python的int在32位系统上通常支持更大的范围,但这里我们模拟32位有符号整数的范围
Go
package main

import (
	"fmt"
	"regexp"
	"strconv"
	"math"
)

func myAtoi(s string) int {
	// 去除前导空格
	s = strings.TrimSpace(s)
	
	// 定义正则表达式,匹配整数部分(包括正负号)
	re := regexp.MustCompile(`^[+-]?\d+`)
	match := re.FindString(s)
	
	if match == "" {
		// 没有找到匹配项,返回0
		return 0
	}
	
	// 提取匹配的整数部分并转换为int64
	num, _ := strconv.ParseInt(match, 10, 64)
	
	// 处理溢出情况
	if num > math.MaxInt32 {
		return math.MaxInt32
	}
	if num < math.MinInt32 {
		return math.MinInt32
	}
	
	// 转换为int类型并返回
	return int(num)
}

func main() {
	// 示例调用
	fmt.Println(myAtoi("   -42"))
}

注意:Go的示例中,strings包需要被导入,但我没有在函数定义中显示它,因为strings.TrimSpace通常用于处理字符串,而不是在myAtoi函数内部。

复杂度分析

  • 时间复杂度:O(n),其中n是字符串s的长度。正则表达式匹配的时间复杂度通常与字符串长度成线性关系。
  • 空间复杂度:O(1),虽然正则表达式匹配过程中可能需要使用额外的空间来存储匹配结果,但这个空间的大小与输入规模无关,因此可以认为是常数空间。

总结

方式优点缺点时间复杂度空间复杂度
方式一逻辑清晰,对输入有明确的处理流程代码较长,实现相对复杂O(n)O(1)
方式二代码简洁,易于理解依赖正则表达式库,可能不是最优解O(n)O(1)

相似题目

相似题目难度链接
leetcode 13. 罗马数字转整数中等LeetCode 13
leetcode 48. 旋转图像中等LeetCode 48
leetcode 58. 最后一个单词的长度简单LeetCode 58

欢迎一键三连(关注+点赞+收藏),技术的路上一起加油!!!代码改变世界

  • 关于我:阿里非典型程序员一枚 ,记录在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名),回复暗号,更能获取学习秘籍和书籍等

  • —⬇️欢迎关注下面的公众号:进朱者赤,认识不一样的技术人。⬇️—

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1845422.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

蔚来汽车AI算法工程师,如何理解注意力?

大家好啊&#xff0c;我是董董灿。 今天分享一个上海蔚来汽车的AI算法岗位面试经验总结帖&#xff0c;面试岗位为算法工程师。 这次面试提到的问题&#xff0c;除了与实习相关内容和反问之外&#xff0c;面试官总共问了8个问题&#xff0c;主要集中在深度学习基础概念的理解上…

裁员裁到大动脉,是一种什么体验!

大家好啊&#xff0c;我是董董灿。 降本增效是每个当老板的人都喜欢挂在嘴边的口头禅&#xff0c;尤其是行业不景气&#xff0c;公司发展遇到瓶颈的时候。 大部分公司降本增效的手段其实非常相似&#xff0c;比较容易实施的手段也就那几种。 要么搞设备自动化和流程自动化&a…

Ubuntu 22.04.1 安装ubuntu有道词典时错误发生

1. Ubuntu环境版本 Linux lipan-Precision-T1700 6.5.0-26-generic #26~22.04.1-Ubuntu SMP PREEMPT_DYNAMIC Tue Mar 12 10:22:43 UTC 2 x86_64 x86_64 x86_64 GNU/Linux 2. 有道词典 下载ubuntu系统的deb安装包。 网易有道翻译-支持文本翻译、文档翻译、AIBox英文写作、智…

LoRa126X系列LoRa模块:专为物联网设计而生

LoRa126X是思为无线研发的一款应用于物联网应用的LoRa 前端模块系列&#xff0c;采用 Semtech 公司的 SX1262和SX1268 芯片。该系列模块具有小体积、低功耗&#xff0c;高灵敏度等特点&#xff0c;并且严格遵循无铅工艺生产和测试流程&#xff0c;符合 RoHS 和 Reach 环保标准。…

重学java 80.Junit单元测试

我总是着急的解释我自己&#xff0c;却忘了厚爱无需多言 —— 24.6.21 一、Junit介绍 1.概述 Junit是一个单元测试框架,可以代替main方法去执行其他的方法 2.作用 可以单独执行一个方法,测试该方法是否能跑通 3.注意 Junit是第三方工具,所以使用之前需要导入jar包 二、J…

HTML和CSS基础(一)

前言 HTML&#xff08;HyperText Markup Language&#xff09;是一种用于创建网页的标准标记语言。它由各种标签组成&#xff0c;这些标签定义了网页的结构和内容。HTML的早期形式诞生于1989年&#xff0c;由CERN的物理学家Tim Berners-Lee发明&#xff0c;最初用于在科学家之…

Laravel - excel 导入数据

在Laravel中&#xff0c;可以使用maatwebsite/excel这个库来处理Excel文件的导入。 1.用命令行窗口打开项目根目录&#xff0c;使用 Composer 安装 maatwebsite/excel composer require maatwebsite/excel --ignore-platform-reqs 在你的config/app.php文件中注册服务提供者&…

二轴机器人大米装箱机:推动行业持续发展

随着科技的不断发展&#xff0c;机器人技术已经深入到各行各业&#xff0c;为传统生产带来了巨大的变革。其中&#xff0c;二轴机器人大米装箱机以其独特的技术特点和应用价值&#xff0c;正在引领大米包装行业的新潮流。 二轴机器人大米装箱机采用了先进的机械臂设计&#xff…

初阶 《数组》 2. 二维数组的创建和初始化

2. 二维数组的创建和初始化 2.1 二维数组的创建 //数组创建 int arr[3][4]; char arr[3][5]; double arr[2][4];2.2 二维数组的初始化 //数组初始化 int arr[3][4] {1,2,3,4}; int arr[3][4] {{1,2},{4,5}}; int arr[][4] {{2,3},{4,5}};//二维数组如果有初始化&#xff…

Wine 安装GDI

如上发图&#xff0c;安装的时候可能出现错误&#xff1a; Cannot find cabextract. Please install it (e.g. sudo apt-get install cabext 解决方案&#xff1a; 终端执行&#xff1a; sudo apt-get update sudo apt-get install cabextract

使用 MLRun 和 MinIO 设置开发机器

MLOps 之于机器学习&#xff0c;就像 DevOps 之于传统软件开发一样。两者都是一组旨在改善工程团队&#xff08;开发或 ML&#xff09;和 IT 运营 &#xff08;Ops&#xff09; 团队之间协作的实践和原则。目标是使用自动化来简化开发生命周期&#xff0c;从规划和开发到部署和…

来聊聊redis文件事件驱动的设计

写在文章开头 近期团队安排变得比较紧急&#xff0c;关于redis系列的更新相对放缓一些&#xff0c;而我们今天要讨论的就是redis中关于事件模型的设计&#xff0c;我们都知道redis通过单线程实现高效的网络IO处理&#xff0c;本文会从源码的角度来讲解一下redis中文件事件驱动…

国内怎样使用GPT4 turbo

GPT是当前最为熟知的大模型&#xff0c;它优越的性能一直遥遥领先于其它一众厂商&#xff0c;然而如此优秀的AI在中国境内却是无法正常使用的。本文将告诉你4种使用gpt4的方法&#xff0c;让你突破限制顺利使用。 官方售价是20美元/月&#xff0c;40次提问/3小时&#xff0c;需…

CSDN自定义模块全攻略,DIY系统原有样式打造专属个性化主页!

个人主页&#xff1a;学习前端的小z 个人专栏&#xff1a;HTML5和CSS3悦读 本专栏旨在分享记录每日学习的前端知识和学习笔记的归纳总结&#xff0c;欢迎大家在评论区交流讨论&#xff01; 文章目录 &#x1f4af;如何通过HTMLCSS自定义模板diy出自己的个性化csdn主页&#x…

Developer Day北京站倒计时三天|请查收您的参会指南!

MongoDB Developer Day 6/22北京站 倒计时3天&#xff0c;期待您的出席&#xff01; 请收藏您的行前温馨贴士⬇️ MongoDB Developer Day 专为开发者打造的动手实操工作坊和模型设计优化专场&#xff01; 本活动针对开发者和产品负责人设计&#xff0c;旨在学习NoSQL数据建…

同三维T80004EH-N HDMI高清NDI编码器

1路HDMI 1路3.5音频输入,支持NDI 产品简介&#xff1a; 同三维T80004EH-N 高清HDMI编码器是专业的NDI高清音视频编码产品&#xff0c;该产品支持1路高清HDMI音视频采集功能&#xff0c;1路3.5MM独立音频接口采集功能。编码输出双码流H.265/H.264格式&#xff0c;音频MP3/AAC格…

初阶 《数组》 3. 数组越界

3. 数组越界 数组的下标是有范围限制的。 数组的下规定是从0开始的&#xff0c;如果数组有n个元素&#xff0c;最后一个元素的下标就是n-1。 所以数组的下标如果小于0&#xff0c;或者大于n-1&#xff0c;就是数组越界访问了&#xff0c;超出了数组合法空间的访问。 C语言本身…

VUE面试题汇总(九)

之间联系&#xff08;Model 和 ViewModel 的双向数据绑定&#xff09; 解析&#xff1a; MVVM 是 Model-View-ViewModel 的缩写。MVVM 是一种设计思想。Model 层代表数据模型&#xff0c;也可以在 Model 中定义数据修改和操作的业务逻辑&#xff1b;View 代表 UI 组件&#xf…

种子流媒体服务器TorrServer

什么是 TorrServer &#xff1f; TorrServer 是一个允许用户在线查看种子而无需预先下载文件的程序。 TorrServer 的核心功能包括缓存种子以及通过 HTTP 协议进行后续数据传输&#xff0c;允许根据系统参数和用户的互联网连接速度调整缓存大小。 软件特点 缓存流媒体本地和远程…

金蝶云星空与MES系统深度集成对接案例全公开

项目背景 深圳市某自动化设备有限公司&#xff0c;自2006年成立以来&#xff0c;一直专注于高端精密自动化设备的研发、生产与销售。作为一家高科技企业&#xff0c;公司依托深圳这一经济特区的地理优势&#xff0c;构建了覆盖全国的服务网络&#xff0c;并拥有两个先进的生产…