怒刷LeetCode的第26天(Java版)

news2025/1/23 17:51:48

第一题

题目来源

64. 最小路径和 - 力扣(LeetCode)

题目内容

解决方法

方法一:动态规划

可以使用动态规划来解决这个问题。

  1. 首先创建一个与网格大小相同的二维数组dp,用于存储从起点到每个位置的最小路径和。
  2. 然后初始化dp[0][0] = grid[0][0],表示起点的最小路径和为起点的值。
  3. 接下来进行动态规划的遍历,遍历顺序可从起点开始按行或按列遍历。对于每个位置dp[i][j],其最小路径和为grid[i][j]加上它左边或上边位置的最小路径和的较小值。

复杂度分析:

时间复杂度:

  • 遍历网格中的每个元素,所以需要进行两层嵌套循环,总共遍历次数为m * n,其中m为网格的行数,n为网格的列数。
  • 在每次循环中,执行常数时间的操作,将上方和左方的最小路径和与当前位置的值相加,然后取最小值。
  • 因此,整体的时间复杂度为O(m * n)。

空间复杂度:

  • 创建了一个与网格大小相同的二维数组dp,用于存储从起点到每个位置的最小路径和。因此需要额外的空间来存储这些最小路径和。
  • dp数组的大小为m * n,与原始网格的大小相同。
  • 所以,空间复杂度为O(m * n)。

综上所述,该算法的时间复杂度为O(m * n),空间复杂度为O(m * n)。

LeetCode运行结果:

第二题

题目来源

65. 有效数字 - 力扣(LeetCode)

题目内容

解决方法

方法一:有限状态自动机

可以使用有限状态自动机(DFA)来解决该问题。我们需要设计合适的状态集合以及状态转移规则。其中,状态包括以下几种:

起始空格状态:在这个状态下,前面的空格已经被忽略。 符号位状态:在这个状态下,可能出现 + 或 -。 整数状态:在这个状态下,输入了数字 0-9。 小数点状态:在这个状态下,已经输入了小数点。 小数状态:在这个状态下,已经输入了数字并带有小数点。 幂符号状态:在这个状态下,出现了字符 e 或 E。 幂符号后的符号位状态:在这个状态下,出现了符号位 + 或 -。 幂符号后的整数状态:在这个状态下,输入了数字 0-9。 终止状态:如果当前输入的字符不合法或者最后一步能够到达终止状态,则说明输入是一个合法的数字。

class Solution {
public boolean isNumber(String s) {
    // 定义有限状态自动机
    Map<State, Map<CharType, State>> transfer = new HashMap<>(); 
    transfer.put(State.STATE_INITIAL, new HashMap<>() {{
        put(CharType.CHAR_SPACE, State.STATE_INITIAL); // 起始空格状态
        put(CharType.CHAR_NUMBER, State.STATE_INTEGER); // 整数状态
        put(CharType.CHAR_SIGN, State.STATE_SIGN); // 符号位状态
        put(CharType.CHAR_POINT, State.STATE_POINT_WITHOUT_INT); // 小数点状态(前面没有数字)
    }});
    transfer.put(State.STATE_SIGN, new HashMap<>() {{
        put(CharType.CHAR_NUMBER, State.STATE_INTEGER); // 整数状态
        put(CharType.CHAR_POINT, State.STATE_POINT_WITHOUT_INT); // 小数点状态(前面没有数字)
    }});
    transfer.put(State.STATE_INTEGER, new HashMap<>() {{
        put(CharType.CHAR_NUMBER, State.STATE_INTEGER); // 整数状态
        put(CharType.CHAR_EXP, State.STATE_EXP); // 幂符号状态
        put(CharType.CHAR_POINT, State.STATE_POINT); // 小数状态
        put(CharType.CHAR_SPACE, State.STATE_END); // 终止状态
    }});
    transfer.put(State.STATE_POINT, new HashMap<>() {{
        put(CharType.CHAR_NUMBER, State.STATE_FRACTION); // 小数状态
        put(CharType.CHAR_EXP, State.STATE_EXP); // 幂符号状态
        put(CharType.CHAR_SPACE, State.STATE_END); // 终止状态
    }});
    transfer.put(State.STATE_POINT_WITHOUT_INT, new HashMap<>() {{
        put(CharType.CHAR_NUMBER, State.STATE_FRACTION); // 小数状态
    }});
    transfer.put(State.STATE_FRACTION, new HashMap<>() {{
        put(CharType.CHAR_NUMBER, State.STATE_FRACTION); // 小数状态
        put(CharType.CHAR_EXP, State.STATE_EXP); // 幂符号状态
        put(CharType.CHAR_SPACE, State.STATE_END); // 终止状态
    }});
    transfer.put(State.STATE_EXP, new HashMap<>() {{
        put(CharType.CHAR_NUMBER, State.STATE_EXP_NUMBER); // 幂符号后的整数状态
        put(CharType.CHAR_SIGN, State.STATE_EXP_SIGN); // 幂符号后的符号位状态
    }});
    transfer.put(State.STATE_EXP_SIGN, new HashMap<>() {{
        put(CharType.CHAR_NUMBER, State.STATE_EXP_NUMBER); // 幂符号后的整数状态
    }});
    transfer.put(State.STATE_EXP_NUMBER, new HashMap<>() {{
        put(CharType.CHAR_NUMBER, State.STATE_EXP_NUMBER); // 幂符号后的整数状态
        put(CharType.CHAR_SPACE, State.STATE_END); // 终止状态
    }});
    transfer.put(State.STATE_END, new HashMap<>() {{
        put(CharType.CHAR_SPACE, State.STATE_END); // 终止状态
    }});
    
    // 根据定义的状态转移规则,判断该字符串是否为有效数字
    int length = s.length();
    State state = State.STATE_INITIAL;
    for (int i = 0; i < length; i++) {
        CharType type = toCharType(s.charAt(i));
        if (!transfer.get(state).containsKey(type)) { // 当前输入不合法
            return false;
        }
        state = transfer.get(state).get(type); // 进入下一个状态
    }
    // 最后一步能够到达终止状态说明该字符串是一个有效数字
    return state == State.STATE_INTEGER || state == State.STATE_POINT ||
            state == State.STATE_FRACTION || state == State.STATE_EXP_NUMBER ||
            state == State.STATE_END;
}

// 定义字符类型枚举类
enum CharType {
    CHAR_NUMBER,
    CHAR_EXP,
    CHAR_POINT,
    CHAR_SIGN,
    CHAR_SPACE,
    CHAR_ILLEGAL
}

// 判断字符类型
private CharType toCharType(char ch) {
    if (ch >= '0' && ch <= '9') {
        return CharType.CHAR_NUMBER;
    } else if (ch == 'e' || ch == 'E') {
        return CharType.CHAR_EXP;
    } else if (ch == '.') {
        return CharType.CHAR_POINT;
    } else if (ch == '+' || ch == '-') {
        return CharType.CHAR_SIGN;
    } else if (ch == ' ') {
        return CharType.CHAR_SPACE;
    } else {
        return CharType.CHAR_ILLEGAL;
    }
}

// 定义状态枚举类
enum State {
    STATE_INITIAL,
    STATE_INTEGER,
    STATE_POINT,
    STATE_POINT_WITHOUT_INT,
    STATE_FRACTION,
    STATE_EXP,
    STATE_EXP_SIGN,
    STATE_EXP_NUMBER,
    STATE_SIGN,
    STATE_END
}
}

复杂度分析:

  • 时间复杂度为O(n),其中n表示字符串的长度。算法需要对字符串中的每个字符进行遍历,因此时间复杂度与字符串的长度成正比。
  • 空间复杂度为O(1),因为算法只使用了常数个变量和一个固定大小的哈希表(状态转移规则),不随输入规模的增加而增加额外的空间消耗。因此,算法的空间复杂度是常数级别的。

LeetCode运行结果:

方法二:正则表达式

除了有限状态自动机,还可以使用正则表达式来判断一个字符串是否为有效数字。Java中提供了一个函数matches(String regex)来判断一个字符串是否能够匹配指定的正则表达式。

class Solution {
public boolean isNumber(String s) {
    // 使用正则表达式匹配字符串
    return s.trim().matches("[+-]?(?:\\d+\\.?\\d*|\\.\\d+)(?:[Ee][+-]?\\d+)?");
}
}

复杂度分析:

该算法的时间复杂度为O(1),因为使用了Java内置函数,时间复杂度是固定的。空间复杂度为O(1),因为没有使用额外的空间。

需要注意的是,使用正则表达式虽然代码简单易懂,但是性能可能不如有限状态自动机。正则表达式求解的时间复杂度是O(n),其中n是字符串长度,适用于一些简单的模式匹配。对于复杂模式,引擎会消耗非常多的时间和空间,建议在使用时注意性能问题。

LeetCode运行结果:

第三题

题目来源

66. 加一 - 力扣(LeetCode)

题目内容

解决方法

方法一:从最后一位向前遍历

该方法从数组的最后一位开始向前遍历,将当前位加一,如果加一后不需要进位,则直接返回结果。如果加一后需要进位,则将当前位置为0,并继续处理前一位。如果所有位都需要进位,则返回新的结果数组,长度为原数组长度加一,首位为1,其余位为0。

class Solution {
public int[] plusOne(int[] digits) {
    int n = digits.length;
    
    // 从最后一位开始向前遍历
    for (int i = n - 1; i >= 0; i--) {
        // 当前位加一
        digits[i]++;
        
        // 如果当前位加一后仍然小于10,没有进位,直接返回结果
        if (digits[i] < 10) {
            return digits;
        }
        
        // 有进位,当前位变为0,继续处理前一位
        digits[i] = 0;
    }
    
    // 若所有位都有进位,则数组长度需要增加1,首位为1,后面全是0
    int[] result = new int[n + 1];
    result[0] = 1;
    
    return result;
}

}

复杂度分析:

  • 时间复杂度:O(n),其中n为数组的长度。在最坏情况下,需要遍历整个数组一次。
  • 空间复杂度:O(n),需要创建一个新的结果数组,长度可能为n+1。

LeetCode运行结果:

方法二:数学运算

还可以使用数学运算来实现加一操作。

  1. 首先将数组最后一位加一,记录进位carry。
  2. 从数组倒数第二位开始,将当前位加上进位carry,并更新进位carry。
  3. 如果进位carry为0,则无需继续处理,直接返回结果数组。
  4. 如果进位carry不为0,则继续向前处理前一位。
  5. 如果处理完所有位后,进位carry仍然不为0,说明需要扩展结果数组的长度,将原数组复制到新的结果数组中,并在首位插入进位carry。
class Solution {
public int[] plusOne(int[] digits) {
    int n = digits.length;
    int carry = 1; // 进位初始化为1
    
    // 从数组最后一位开始计算
    for (int i = n - 1; i >= 0; i--) {
        digits[i] += carry; // 当前位加上进位
        carry = digits[i] / 10; // 计算新的进位
        digits[i] %= 10; // 当前位取模,得到个位数

        // 如果进位为0,说明不需要再进位,直接返回结果数组
        if (carry == 0) {
            return digits;
        }
    }
    
    // 若所有位都有进位,则数组长度需要增加1,首位为进位carry,后面全是0
    int[] result = new int[n + 1];
    result[0] = carry;
    System.arraycopy(digits, 0, result, 1, n);
    
    return result;
}

}

复杂度分析:

  • 时间复杂度:遍历数组需要线性时间O(n),其中每个元素只进行一次常数级别的数学运算,所以总体时间复杂度也是O(n)。
  • 空间复杂度:除了原数组外,需要在进位的情况下创建一个新数组来存储结果,所以空间复杂度为O(n)。

LeetCode运行结果:

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

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

相关文章

shiro反序列化和log4j

文章目录 安装环境shiro漏洞验证log4j 安装环境 进入vulhb目录下的weblogic&#xff0c;复现CVE-2018-2894漏洞&#xff1a; cd /vulhub/shiro/CVE-2010-3863查看docker-compose的配置文件&#xff1a; cat docker-compose.yml如图&#xff0c;里面有一个镜像文件的信息和服…

136.【JUC并发编程_02】

JUC并发编程 (四)、共享模型之管程1.wait notify(1).小故事_为什么需要wait(2).wait notify 的工作原理(3).API介绍 2.wait notify 的正确使用步骤 ⭐(1).sleep 和 wait 的区别(2).步骤1_产生的问题(3).步骤2_wait notify 改进产生问题(4).步骤3_产生叫错人问题 (虚假唤醒)(5).…

opencv安装成功之后运行代码还是出错

错误提示 Traceback (most recent call last): File "F:\download\55957_人工智能基础与应用&#xff08;微课版&#xff09;_源代码\OpenCV\camera.py", line 4, in <module> import cv2 File "F:\software\anaconda\envs\tensorflow\cv2\__init__.py&q…

Unity 热更新技术 | (一) 热更新的基本概念原理及主流热更新方案介绍

&#x1f3ac; 博客主页&#xff1a;https://xiaoy.blog.csdn.net &#x1f3a5; 本文由 呆呆敲代码的小Y 原创&#xff0c;首发于 CSDN&#x1f649; &#x1f384; 学习专栏推荐&#xff1a;Unity系统学习专栏 &#x1f332; 游戏制作专栏推荐&#xff1a;游戏制作 &…

什么是强缓存、协商缓存?

为了减少资源请求次数,加快资源访问速度,浏览器会对资源文件如图片、css文件、js文件等进行缓存,而浏览器缓存策略又分为强缓存和协商缓存,什么是强缓存?什么是协商缓存?两者之间的区别又是什么?接下来本文就带大家深入了解这方面的知识。 强缓存 所谓强缓存,可以理解…

声音生成评价项目AudioLDM_eval项目配置过程

文章目录 引言正文问题一&#xff1a;模型下载不了问题二 TypeError: pad_center() takes 1 positional argument but 2 were given问题三 AttributeError: module numpy has no attribute complex. 结果 引言 对于生成的声音&#xff0c;如何进行评价&#xff0c;一般是通过计…

Matlab之查询子字符串在字符串中的起始位置函数strfind

一、功能 strfind函数用于在一个字符串中查找指定的子字符串&#xff0c;并返回子字符串在字符串中的起始位置。 二、语法 indices strfind(str, pattern) 其中&#xff0c;str是要进行查找的字符串&#xff0c;pattern是要查找的子字符串。 函数会返回一个由子字符串在字…

网络与信息安全基础知识 (软件设计师笔记)

&#x1f600;前言 在当今世界&#xff0c;我们见证了科技&#xff0c;特别是网络技术的繁荣发展&#xff0c;这种发展不仅让我们的生活变得更加便捷&#xff0c;但也带来了一系列的安全问题。网络安全不仅关系到每一个上网的个人&#xff0c;更是关乎到国家的安全和社会的稳定…

大数据要怎么样学才可以到企业级实战

大数据在企业级实战中扮演着重要角色&#xff0c;因此掌握大数据技术和应用是非常有价值的。下面将详细介绍学习大数据并达到企业级实战水平的步骤和方法。 一、基础知识准备 1. 数据基础知识&#xff1a;了解数据的概念、类型、结构等基本概念&#xff0c;并熟悉常见的数据处…

Edge 无法登录/同步问题【一招搞定】

目录 前言 一、打开 Edge 浏览器显示未同步&#xff0c;点击同步无效 二、Edge 登录报错 0x801901f4 或 0x80190001 解决方法 2.1 报错 0x801901f4 解决方法 2.1.0 Edge 登陆报错图示 2.1.1 添加 Edge 推荐的 DNS 地址 2.1.2 重新登录 Edge 账号成功 2.2 报错 0x801…

第四章 树和二叉树

第四章 树和二叉树 树的基本概念树的概念树的相关术语 二叉树二叉树基本概念二叉树的性质 二叉树的存储结构二叉树的顺序存储结构二叉树的链式存储结构 二叉树的遍历二叉树遍历的递归实现二叉树的层次遍历二叉树遍历的非递归实现 树和森林树的存储结构树、森林与二叉树的关系树…

【低代码开发】:低代码开发助力应用创新

低代码开发&#xff1a;加速应用开发的未来趋势 引言什么是低代码以及功能特点&#xff1f;什么是低代码开发&#xff1f;低代码平台的特点和功能低代码平台的应用场景和优势低代码的优点低代码的缺点低代码平台项目开发流程选择和实施低代码平台 低代码未来的发展趋势低代码平…

Java基础(变量篇)

变量是Java程序中基本的存储单元&#xff0c;变量名有三个基本要素&#xff1a;数据类型、变量名和值。变量名是一块内存单元的名称&#xff0c;就像门牌号一样&#xff0c;通过变量可以找到它表示的内存单元&#xff0c;并对这块内存单元进行操作。在Java中变量必须声明后使用…

英国/法国/意大利/德国/西班牙,电动交通设备配件等相关政策更新

产品安全 合规政策更新&#xff01; 首先请看邮件内容 尊敬的卖家&#xff1a; 您好&#xff01; 我们此次联系您是因为您正在销售需要审批流程的商品。为此&#xff0c;亚马逊正在实施审批流程&#xff0c;以确认我们网站上提供的商品类型须符合指定的认证标准。要在亚马逊…

Cesium小技巧:快速打开API文档

学习Cesium.js的人&#xff0c;肯定经常看官方示例&#xff0c; 网址如下&#xff1a; https://sandcastle.cesium.com/ 有个小技巧&#xff0c;可以快速打开具体类的API文档 在示例中&#xff0c;双击具体类名或方法名&#xff0c;会出现一个提示框 单击或右键菜单-在新标…

Linux系统编程:Linux基础

Linux基本使用 安装 这就不说了网上很多教程&#xff0c;随便一个Linux版本都可以&#xff0c;因为命令都差不多。 检查是否有网络 检查网络有以下几个推测方法&#xff0c;如果没有网络的话就按照这几个可能出现问题的地方对症下药寻找解决办法即可&#xff1a; ctrl al…

蓝桥杯基础---切面条

切面条 一根高筋拉面&#xff0c;中间切一刀&#xff0c;可以得到2根面条。 如果先对折1次&#xff0c;中间切一刀&#xff0c;可以得到3根面条。 如果连续对折2次&#xff0c;中间切一刀&#xff0c;可以得到5根面条。 那么&#xff0c;连续对折10次&#xff0c;中间切一刀…

MySQL数据库安装和介绍

一.数据库的介绍 1.1.MySQL数据库是什么&#xff1f; 一款深受欢迎的开源关系型数据库&#xff0c;Oracle旗下的产品&#xff0c;遵守GPL协议&#xff0c;可以免费使用与修改。 官网&#xff1a;MySQL 特点&#xff1a; 性能卓越、服务稳定 开源、无版权限制、成本低 …

优雅而高效的JavaScript——函数柯里化

&#x1f642;博主&#xff1a;小猫娃来啦 &#x1f642;文章核心&#xff1a;优雅而高效的JavaScript——函数柯里化 文章目录 前言函数柯里化的概念和原理介绍函数柯里化的优点和应用场景函数柯里化的具体实现方式和实例分析函数柯里化在实际项目中的应用函数柯里化与函数式编…

数据结构:链表(1)

顺序表的优缺点 缺点&#xff1a; 1.插入数据必须移动其他数据&#xff0c;最坏情况下&#xff0c;就是插入到0位置。时间复杂度O(N) 2.删除数据必须移动其他数据&#xff0c;最坏情况下&#xff0c;就是删除0位置。时间复杂度O(N) 3.扩容之后&#xff0c;有可能会浪费空间…