题目:请你来实现一个 myAtoi(string s) 函数,使其能将字符串转换成一个 32 位有符号整数。

01、手动处理每个字符法

最简单的方法永远是脑海中第一个想到的方法,也是最暴力的方法,而这一题我们只需要安装题目要求一个一个字符处理即可,其实整个解法相当简单。

我们梳理一下解题思路,大致可分为以下三步:

(1)处理开头空格,通过while循环把开头的所有空格都去除掉;

(2)处理正负符号,判断是否包含+/-号,并做下标记;

(3)处理数字,首先判断当前字符是否为数值,如果是则计算出其值,并且检测是否溢出,在未溢出的情况下,通过x*10+digit的方式累计结果;

(4)最后根据正负号返回正确结果。

具体代码如下:

//解法 1:手动处理每个字符(经典解法)
public static int MyAtoi1(string s)
{
//结果
var result = 0;
//当前处理字符索引
var index = 0;
//标记正负数
var sign = 1;
//字符串长度
var length = s.Length;
//去除开头的空格
while (index < length && s[index] == ' ')
{
//处理下一个字符
index++;
}
//处理正负符号
if (index < length && (s[index] == '+' || s[index] == '-'))
{
//标记正负数
sign = s[index] == '-' ? -1 : 1;
//处理下一个字符
index++;
}
//转换数字字符为数字
while (index < length && char.IsDigit(s[index]))
{
//计算当前字符数值
var digit = s[index] - '0';
//检查是否溢出
if (result > (int.MaxValue - digit) / 10)
{
return sign == 1 ? int.MaxValue : int.MinValue;
}
//累积当前字符至结果中
result = result * 10 + digit;
//处理下一个字符
index++;
}
//返回结果
return sign * result;
}

02、正则表达式法

还有一种比较简洁的方式,可以直接通过正则表达式匹配出满足条件的字符,然后再通过BigInteger.TryParse进行类型转换,之所以选择BigInteger类型是因为其他值类型都可能导致溢出情况。

根据上一题的解题思路,我们来尝试一步一步用正则表达式实现。

首先需要匹配开头的空白字符。

^:表示一个锚点,匹配字符串的开头;

\s:表示特殊字符,匹配空白字符,包括空格、制表符、换行符等;

*:表示量词,表示其前面元素可以出现零次或多次;

因此可以通过^\s*来实现字符串开头的空格处理。

[]:定义一个字符类,表示匹配方括号中的任意一个字符;

+-:表示加号和减号,即正数和负数符号;

?:表示量词,表示其前面元素可以出现零次或一次;

因此可以通过[+-]?来实现匹配正负号。

\d:表示匹配一个数字字符,范围为0-9;

+:表示量词,表示其前面元素可以出现一次或多次;

因此可以通过\d+来实现连续数字字符匹配。

下面看看具体实现代码:

//解法 2:正则表达式法
public static int MyAtoi2(string s)
{
//使用正则表达式匹配符合要求的部分
//^\s*:表示匹配字符串开头的零个或多个空白字符(空格、制表符等)。
//[+-]?:表示符号(+ 或 -)可选。
//\d+:表示一个或多个数字。
var match = System.Text.RegularExpressions.Regex.Match(s, @"^\s*[+-]?\d+");
//匹配成功,并且可以转换为数值
if (match.Success && BigInteger.TryParse(match.Value, out var result))
{
//大于int最大值
if (result > int.MaxValue)
{
return int.MaxValue;
}
//小于int最小值
if (result < int.MinValue)
{
return int.MinValue;
}
//返回结果
return (int)result;
}
return 0;
}

03、状态机法

此题还有一种经典解法,即状态机法,状态机的核心思想是在一组有限的状态中,并通过输入触发状态之间的转移。

我们以此题为例,假设在处理字符串的过程中,一直存在一个状态state,而每次处理的当前字符则可以触发当状态state转移到下一个状态state1,如此我们只需要枚举出所有state和当前字符关于state1的映射关系即可。

我们可以建立如下图所示状态机:

如上图,如果当前state为“开始”,并且当前字符为“空格”,则state1为“开始”,如果当前字符为“数字”,则state1为“处理数字”,以此类推,而state1则会绝对当前字符具体的处理方法。

我们也可以用以下表格来表示这个状态机:

有了状态机状态关系映射表,我们就可以进行代码实现了,其逻辑也很简单,大致分为以下四步:

(1)构建状态机状态表;

(2)获取当前字符对应状态;

(3)通过状态转移确定当前字符处理逻辑;

(4)对要处理的字符串进行遍历处理得到最终结果;

具体实现代码如下:

//解法 3:状态机法
public int MyAtoi3(string s)
{
Automaton automaton = new Automaton();
return automaton.Atoi(s);
}
public class Automaton
{
//0:"开始"状态
private const int Start = 0;
//1:"标记符号"状态
private const int Signed = 1;
//2:"处理数字"状态
private const int InNumber = 2;
//3:"结束"状态
private const int End = 3;
//符号:1为正数,0为负数
private int _sign = 1;
//数值结果
private long _answer = 0;
//记录当前处理状态
private int _state = Start;
//状态表
private readonly Dictionary<int, int[]> _table = new Dictionary<int, int[]>()
{
{Start,new int[]{ Start, Signed, InNumber, End}},
{Signed,new int[]{ End, End, InNumber, End}},
{InNumber,new int[]{ End, End, InNumber, End}},
{End,new int[]{ End, End, End, End}},
};
//处理当前字符
private void Handle(char c)
{
//获取当前状态
var currentState = GetState(c);
//转移状态
_state = _table[_state][currentState];
switch (_state)
{
//处理数字
case InNumber:
_answer = _answer * 10 + c - '0';
//溢出判断
_answer = _sign == 1 ? Math.Min(_answer, int.MaxValue) : Math.Min(_answer, -(long)int.MinValue);
break;
//处理正负号
case Signed:
_sign = c == '+' ? 1 : -1;
break;
case Start:
case End:
break;
}
}
//获取当前字符对应状态
private static int GetState(char c)
{
//空格
if (char.IsWhiteSpace(c))
{
return Start;
}
//正负号
if (c == '+' || c == '-')
{
return Signed;
}
//数字
if (char.IsDigit(c))
{
return InNumber;
}
//其他
return End;
}
//字符串转换为整数
public int Atoi(string s)
{
var length = s.Length;
for (int i = 0; i < length; ++i)
{
Handle(s[i]);
}
return (int)(_sign * _answer);
}
}

:测试方法代码以及示例源码都已经上传至代码库,有兴趣的可以看看。https://gitee.com/hugogoos/Planner

LeetCode题集-8 - 字符串转换整数 (atoi)的更多相关文章

  1. 字符串转换整数 (atoi) C++实现 java实现 leetcode系列(八)

    字符串转换整数 (atoi) java实现 C++实现 请你来实现一个 atoi 函数,使其能将字符串转换成整数. 首先,该函数会根据需要丢弃无用的开头空格字符,直到寻找到第一个非空格的字符为止. 当 ...

  2. 前端与算法 leetcode 8. 字符串转换整数 (atoi)

    目录 # 前端与算法 leetcode 8. 字符串转换整数 (atoi) 题目描述 概要 提示 解析 解法一:正则 解法二:api 解法二:手搓一个api 算法 传入测试用例的运行结果 执行结果 G ...

  3. LeetCode 8. 字符串转换整数 (atoi)(String to Integer (atoi))

    8. 字符串转换整数 (atoi) 8. String to Integer (atoi) 题目描述 LeetCode LeetCode8. String to Integer (atoi)中等 Ja ...

  4. LeetCode Golang 8. 字符串转换整数 (atoi)

    8. 字符串转换整数 (atoi) 首先,该函数会根据需要丢弃无用的开头空格字符,直到寻找到第一个非空格的字符为止. 当我们寻找到的第一个非空字符为正或者负号时,则将该符号与之后面尽可能多的连续数字组 ...

  5. Leetcode(8)字符串转换整数

    Leetcode(8)字符串转换整数 [题目表述]: 请你来实现一个 atoi 函数,使其能将字符串转换成整数. 首先,该函数会根据需要丢弃无用的开头空格字符,直到寻找到第一个非空格的字符为止. 当我 ...

  6. 17、字符串转换整数 (atoi)

    17.字符串转换整数 (atoi) 请你来实现一个 atoi 函数,使其能将字符串转换成整数. 首先,该函数会根据需要丢弃无用的开头空格字符,直到寻找到第一个非空格的字符为止. 当我们寻找到的第一个非 ...

  7. 8. 字符串转换整数 (atoi)

    8. 字符串转换整数 (atoi) 方法一 import re import math class Solution(object): def myAtoi(self, str): "&qu ...

  8. 每日一题LeetCode 8. 字符串转换整数 (atoi)

    问题描述 请你来实现一个 atoi 函数,使其能将字符串转换成整数. 首先,该函数会根据需要丢弃无用的开头空格字符,直到寻找到第一个非空格的字符为止. 当我们寻找到的第一个非空字符为正或者负号时,则将 ...

  9. leetcode刷题笔记08 字符串转整数 (atoi)

    题目描述 实现 atoi,将字符串转为整数. 在找到第一个非空字符之前,需要移除掉字符串中的空格字符.如果第一个非空字符是正号或负号,选取该符号,并将其与后面尽可能多的连续的数字组合起来,这部分字符即 ...

  10. 【LeetCode】String to Integer (atoi)(字符串转换整数 (atoi))

    这道题是LeetCode里的第8道题. 题目要求: 请你来实现一个 atoi 函数,使其能将字符串转换成整数. 首先,该函数会根据需要丢弃无用的开头空格字符,直到寻找到第一个非空格的字符为止. 当我们 ...

随机推荐

  1. 使用pxe安装ARM服务器(鲲鹏920)遇到的坑

    一.关于PXE获取到IP之后无ACK,无法获取引导文件. 目前ARM服务器基本都是使用UEFI的方式进行引导,我们只需要关注EFI方式引导即可,Legacy引导已经随着时代的发展被扫进历史的垃圾桶. ...

  2. centos7 nginx+php7yum安装

    centos7 nginx+php7yum安装. 一.安装nginx 1.安装yum源 rpm -Uvh http://nginx.org/packages/centos/7/noarch/RPMS/ ...

  3. PC软件开发新体验!用 Blazor Hybrid 打造简洁高效的视频处理工具

    前言 国庆假期各种活动比较多,直到上班才有时间来更新文章~ 不过这两天我还是做了个小玩意(Clipify),起因是想给之前开发来自己用的简单视频剪辑工具 QuickCutSharp 加个功能,不过这个 ...

  4. vant2 自动检查表单验证 -validate

    ref 给 <van-form @submit="onSubmit" ref="form"> 标签 : // 检验手机号是否合格 await thi ...

  5. 删除 设置 获取 cookies 的第三方包 js-cookies

    // 第三方包 js-cookies 是一个操作cookies的包 import Cookies from 'js-cookie' // Cookies.set(key,value) 存值 // Co ...

  6. 鸿蒙NEXT开发声明式UI是咋回事?

    大家好,我是 V 哥,ArkTS 是 HarmonyOS 优选的主力应用开发语言,它在 TypeScript 的基础上进行了扩展,提供了声明式 UI 描述.自定义组件和动态扩展 UI 元素的能力.这些 ...

  7. KubeSphere 社区双周报 | 功能亮点抢“鲜”看 | 2022-09-30

    KubeSphere 从诞生的第一天起便秉持着开源.开放的理念,并且以社区的方式成长,如今 KubeSphere 已经成为全球最受欢迎的开源容器平台之一.这些都离不开社区小伙伴的共同努力,你们为 Ku ...

  8. IO体系

    IO,即in和out,也就是输入和输出,指应用程序和外部设备之间的数据传递,常见的外部设备包括文件.管道.网络连接. Java 中是通过流处理IO 的,那么什么是流? 流(Stream),是一个抽象的 ...

  9. python 爬虫如何爬取动态生成的网页内容

    ---  好的方法很多,我们先掌握一种  --- [背景] 对于网页信息的采集,静态页面我们通常都可以通过python的request.get()库就能获取到整个页面的信息. 但是对于动态生成的网页信 ...

  10. [Linux]学习之路---树梅派4B出现打开文件管理器闪退等问题

    直接控制台运行命令: sudo apt-get install --reinstall pcmanfm 后面的pcmanfm,是一个功能齐全的Linux上的轻量级文件管理器,我自己的记忆方法就是: P ...