![](https://img-blog.csdnimg.cn/e73fb786350c495d9cf2498d1d493445.png?x-oss-process=image/resize,m_fixed,h_224,w_224)
剑指offer
文章平均质量分 65
剑指offer基础算法练习
庄小焱
我是庄小焱,某大厂Java高级工程师、PMP项目管理专家、系统架构设计师(高级)、CSDN博文专家。 博主在支付领域,金融领域、订单履约领域不断学习,同时在博客中分享自己学习知识和相关技术, 欢迎大家和我交流学习,欢迎大家关注我的博客。
展开
-
剑指 Offer 03. 数组中重复的数字
找出数组中重复的数字。在一个长度为 n 的数组 nums 里的所有数字都在 0~n-1 的范围内。数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了几次。请找出数组中任意一个重复的数字。原创 2023-02-12 10:42:43 · 462 阅读 · 1 评论 -
剑指 Offer 04. 二维数组中的查找
由于 (x,y)的初始值分别为(0,m−1),因此y最多能被减少m次,x最多能被增加n次,总搜索次数n+m。如果 matrix[x,y]>target,由于每一列的元素都是升序排列的,那么在当前的搜索矩阵中,所有位于第y 列的元素都是严格大于target的,因此我们可以将它们全部忽略,即将y减少1;如果 matrix[x,y]原创 2023-02-24 19:40:08 · 522 阅读 · 0 评论 -
剑指 Offer 05. 替换空格
由于每次替换从1个字符变成3个字符,使用字符数组可方便地进行替换。建立字符数组地长度为 s 的长度的3倍,这样可保证字符数组可以容纳所有替换后的字符。空间复杂度:O(n)O(n)。额外创建字符数组,长度为 s 的长度的 3 倍。时间复杂度:O(n)O(n)。遍历字符串 s 一遍。原创 2023-02-24 19:49:03 · 547 阅读 · 0 评论 -
剑指 Offer 06. 从尾到头打印链表
【代码】剑指 Offer 06. 从尾到头打印链表。原创 2023-02-23 22:53:31 · 149 阅读 · 0 评论 -
剑指 Offer 07. 重建二叉树
输入某二叉树的前序遍历和中序遍历的结果,请构建该二叉树并返回其根节点。假设输入的前序遍历和中序遍历的结果中都不含重复的数字。在「递归」地遍历某个子树的过程中,我们也是将这颗子树看成一颗全新的树,按照上述的顺序进行遍历。挖掘「前序遍历」和「中序遍历」的性质,我们就可以得出本题的做法。原创 2023-02-12 11:21:05 · 339 阅读 · 0 评论 -
剑指 Offer 08. 二叉树的下一个结点
如果给出的结点无右子节点,且当前结点是其父节点的右子节点,则先要沿着左上方父节点爬树,一直爬到当前结点是其父节点的左子节点为止,返回的就是这个父节点;根据给定输入中的结点指针向父级进行迭代,直到找到该树的根节点;然后根据根节点进行中序遍历,当遍历到和给定树节点相同的节点时,下一个节点就是我们的目标返回节点。step 3:判断该节点是否符合思路中第三点,则迭代向上找父节点,直到迭代的当前节点是父节点的左孩子节点为止,返回该父节点;如果给出的结点无右子节点,且当前结点是其父节点的左子节点,则返回其父节点。原创 2023-02-24 20:28:20 · 165 阅读 · 0 评论 -
剑指 Offer 09. 用两个栈实现队列
【代码】剑指 Offer 09. 用两个栈实现队列。原创 2023-02-23 23:18:25 · 66 阅读 · 0 评论 -
剑指 Offer 10- 斐波那契数列与青蛙跳台阶问题
【代码】剑指 Offer 10- I. 斐波那契数列与青蛙跳台阶问题。原创 2023-02-24 21:13:30 · 77 阅读 · 0 评论 -
剑指 Offer 11. 旋转数组的最小数字
如下图所示,由于重复元素的存在,我们并不能确定numbers[pivot]究竟在最小值的左侧还是右侧,因此我们不能莽撞地忽略某一部分的元素。我们唯一可以知道的是,由于它们的值相同,所以无论 numbers[high]是不是最小值,都有一个它的替代品numbers[pivot],因此我们可以忽略二分查找区间的右端点。在二分查找的每一步中,左边界为low,右边界为high,区间的中点为pivot,最小值就在该区间内。我们考虑数组中的最后一个元素x:在最小值右侧的元素,它们的值一定都小于等于x;原创 2023-02-24 21:39:37 · 441 阅读 · 0 评论 -
剑指 Offer 12. 矩阵中的路径
空间复杂度 O(K): 搜索过程中的递归深度不超过K ,因此系统因函数调用累计使用的栈空间占用 O(K)(因为函数返回后,系统调用的栈空间会释放)。最坏情况下 K=MN,递归深度为MN ,此时系统栈使用 O(MN)的额外空间。时间复杂度 O((3^K)MN) : 最差情况下,需要遍历矩阵中长度为K字符串的所有方案,时间复杂度为 O(3K);返回false :(1) 行或列索引越界或(2) 当前矩阵元素与目标字符不同或(3) 当前矩阵元素已访问过 ((3) 可合并至 (2) )。原创 2023-02-24 23:19:46 · 467 阅读 · 0 评论 -
剑指 Offer 13. 机器人的运动范围
摘要摘要JZ13 机器人的运动范围(地上有一个 rows 行和 cols 列的方格。坐标从 [0,0] 到 [rows-1,cols-1]。一个机器人从坐标 [0,0] 的格子开始移动,每一次只能向左,右,上,下四个方向移动一格,但是不能进入行坐标和列坐标的数位之和大于 threshold 的格子。例如,当 threshold 为 18 时,机器人能够进入方格 [35,37] ,因为 3+5+3+7 = 18。但是,它不能进入方格 [35,38] ,因为 3+5+3+8 = 19。原创 2023-03-02 21:34:51 · 293 阅读 · 0 评论 -
剑指 Offer 14-剪绳子
由于j的取值范围是1到i−1,需要遍历所有的j得到dp[i]的最大值,因此可以得到状态转移方程如下:dp[i]=max(1≤j原创 2023-03-01 22:14:17 · 271 阅读 · 0 评论 -
剑指 Offer 15. 二进制中1的个数
编写一个函数,输入是一个无符号整数(以二进制串的形式),返回其二进制表达式中数字位数为 '1' 的个数。原创 2023-02-18 21:24:41 · 58 阅读 · 0 评论 -
剑指 Offer 16. 数值的整数次方
本题的方法被称为快速幂算法,有递归和迭代两个版本。这篇题解会从递归版本的开始讲起,再逐步引出迭代的版本。当指数n为负数时,我们可以计算 x^(-n)再取倒数得到结果,因此我们只需要考虑n为自然数的情况。原创 2023-03-02 20:46:57 · 281 阅读 · 0 评论 -
剑指 Offer 17. 打印从1到最大的n位数
并且, String 类型的数字的进位操作效率较低,例如 "9999" 至 "10000" 需要从个位到千位循环判断,进位 4 次。观察可知,生成的列表实际上是 n位 0-9 的 全排列 ,因此可避开进位操作,通过递归生成数字的 String 列表。基于分治算法的思想,先固定高位,向低位递归,当个位已被固定时,添加数字的字符串。例如当n=2时(数字范围1−99 ),固定十位为0-9 ,按顺序依次开启递归,固定个位0- 9 ,终止递归并添加数字字符串。实际上,本题的主要考点是大数越界情况下的打印。原创 2023-03-02 21:06:37 · 187 阅读 · 0 评论 -
剑指 Offer 18. 删除链表的节点
【代码】剑指 Offer 18. 删除链表的节点。原创 2023-03-03 14:09:43 · 61 阅读 · 0 评论 -
剑指 Offer 19. 正则表达式匹配
情况二:A[n−1]是多个c中的最后一个(这种情况必须 A[n−1]=c或者 c=′.’),所以A匹配完往前挪一个,B继续匹配,因为可以匹配多个,继续看 A0..n−2和 B0..m−1是否匹配。如果B的最后一个字符是正常字符,那就是看 A[n−1]是否等于 B[m−1],相等则看A0..n−2与 B0..m−2,不等则是不能匹配,这就是子问题。非空串和空正则必不匹配,f[1][0]=...=f[n][0]=false。看:正则串不动,主串前移一个,f[i][j]=f[i−1][j]原创 2023-03-03 19:47:47 · 691 阅读 · 0 评论 -
剑指 Offer 20. 表示数值的字符串
按照字符串从左到右的顺序,定义以下 9 种状态。原创 2023-03-03 20:18:33 · 637 阅读 · 0 评论 -
剑指 Offer 21. 调整数组顺序使奇数位于偶数前面
可始终保证:指针 i左边都是奇数,指针j右边都是偶数。原创 2023-02-18 11:12:07 · 403 阅读 · 0 评论 -
剑指 Offer 22. 链表中倒数第k个节点
我们将第一个指针 fastfast 指向链表的第k+1个节点,第二个指针 slow指向链表的第一个节点,此时指针fast与slow二者之间刚好间隔k个节点。思路与算法:最简单直接的方法即为顺序查找,假设当前链表的长度为n,则我们知道链表的倒数第k个节点即为正数第n−k个节点,此时我们只需要顺序遍历到链表的第 n−k个节点即为倒数第k个节点。我们首先将 slow指向链表的头节点,同时slow与fast同步向后走,当fast指针指向链表的尾部空节点时,则此时返回slow所指向的节点即可。空间复杂度:O(1)。原创 2023-03-03 21:13:53 · 52 阅读 · 0 评论 -
剑指 Offer 23 链表中环的入口结点
慢指针每次只移动一步,而快指针每次移动两步。这样一来,如果在移动的过程中,快指针反过来追上慢指针,就说明该链表为环形链表。随后,slow指针每次向后移动一个位置,而fast指针向后移动两个位置。此时,fast指针已经走完了环的n圈,因此它走过的总距离为 a+n(b+c)+b=a+(n+1)b+nc。有了 a=c+(n−1)(b+c) 的等量关系,我们会发现:从相遇点到入环点的距离加上n−1 圈的环长,恰好等于从链表头部到入环点的距离。根据题意,任意时刻,fast指针走过的距离都为slow指针的2倍。原创 2023-03-03 21:25:42 · 453 阅读 · 0 评论 -
剑指 Offer 24. 反转链表
【代码】剑指 Offer 24. 反转链表。原创 2023-03-03 21:01:40 · 66 阅读 · 0 评论 -
剑指 Offer 25. 合并两个排序的链表
输入两个递增排序的链表,合并这两个链表并使新链表中的节点仍然是递增排序的。原创 2023-02-11 11:15:46 · 79 阅读 · 0 评论 -
剑指 Offer 26. 树的子结构
输入两棵二叉树A和B,判断B是不是A的子结构。(约定空树不是任意一个树的子结构),B是A的子结构, 即 A中有出现和B相同的结构和节点值。原创 2023-02-20 22:42:26 · 188 阅读 · 0 评论 -
剑指 Offer 27. 二叉树的镜像
这是一道很经典的二叉树问题。显然,我们从根节点开始,递归地对树进行遍历,并从叶子节点先开始翻转得到镜像。如果当前遍历到的节点root的左右两棵子树都已经翻转得到镜像,那么我们只需要交换两棵子树的位置,即可得到以root为根节点的整棵子树的镜像。原创 2023-02-20 23:40:11 · 82 阅读 · 0 评论 -
剑指 Offer 28. 对称的二叉树
我们可以实现这样一个递归函数,通过同步移动两个指针的方法来遍历这棵树,p指针和q 指针一开始都指向这棵树的根,随后p右移时,q左移,p左移时,q右移。每次检查当前p和q节点的值是否相等,如果相等再判断左右子树是否对称。因此,该问题可以转化为:两个树在什么情况下互为镜像?如果一个树的左子树与右子树镜像对称,那么这个树是对称的。原创 2023-03-05 21:53:42 · 58 阅读 · 0 评论 -
剑指 Offer 29. 顺时针打印矩阵
输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字。原创 2023-03-05 22:12:05 · 137 阅读 · 1 评论 -
剑指 Offer 30. 包含min函数的栈
对于栈来说,如果一个元素a在入栈时,栈里有其它的元素b, c, d,那么无论这个栈在之后经历了什么操作,只要a在栈中,b, c, d 就一定在栈中,因为在 a 被弹出之前,b, c, d 不会被弹出。因此我们使用一个辅助栈,与元素栈同步插入与删除,用于存储与每个元素对应的最小值。当一个元素要入栈时,我们取当前辅助栈的栈顶存储的最小值,与当前元素比较得出最小值,将这个最小值插入辅助栈中;在任意一个时刻,栈内元素的最小值就存储在辅助栈的栈顶元素中。当一个元素要出栈时,我们把辅助栈的栈顶元素也一并弹出;原创 2023-02-24 19:12:39 · 477 阅读 · 0 评论 -
剑指 Offer 31. 栈的压入、弹出序列
假设压入栈的所有数字均不相等。例如,序列 {1,2,3,4,5} 是某栈的压栈序列,序列 {4,5,3,2,1} 是该压栈序列对应的一个弹出序列,但 {4,3,5,1,2} 就不可能是该压栈序列的弹出序列。每次将 pushed的元素入栈之后,如果栈不为空且栈顶元素与popped的当前元素相同,则将栈顶元素出栈,同时遍历数组 popped,直到栈为空或栈顶元素与 popped的当前元素不同。如果pushed和popped是有效的栈操作序列,则经过所有的入栈和出栈操作之后,每个元素各入栈和出栈一次,栈为空。原创 2023-03-05 21:45:07 · 165 阅读 · 0 评论 -
剑指 Offer 32 - 二叉树的层序遍历
题目要求的二叉树的从上至下打印(即按层打印),又称为二叉树的广度优先搜索(BFS)。BFS通常借助队列的先入先出特性来实现。原创 2023-02-21 21:06:37 · 199 阅读 · 0 评论 -
剑指 Offer 33. 二叉搜索树的后序遍历序列
剑指 Offer 33. 二叉搜索树的后序遍历序列。原创 2023-02-21 22:07:15 · 303 阅读 · 0 评论 -
剑指 Offer 34. 二叉树中和为某一值的路径
注意到本题的要求是,找到所有满足从到某个经过的路径上的节点之和等于目标和的路径。核心思想是对树进行一次遍历,在遍历时记录从根节点到当前节点的路径和,以防止重复计算。原创 2023-02-23 16:20:11 · 311 阅读 · 0 评论 -
剑指 Offer 35. 复杂链表的复制
遍历该链表的过程中,我们检查「当前节点的后继节点」和「当前节点的随机指针指向的节点」的创建情况。注意一个节点可能被多个其他节点指向,因此我们可能递归地多次尝试拷贝某个节点,为了防止重复拷贝,我们需要首先检查当前节点是否被拷贝过,如果已经拷贝过,我们可以直接从哈希表中取出拷贝后的节点的指针并返回即可。对于当前节点,我们首先要进行拷贝,然后我们进行「当前节点的后继节点」和「当前节点的随机指针指向的节点」拷贝,拷贝完成后将创建的新节点的指针返回,即可完成当前节点的两指针的赋值。原创 2023-02-11 13:56:15 · 169 阅读 · 0 评论 -
剑指 Offer 36. 二叉搜索树与双向链表
对于双向循环链表,第一个节点的前驱是最后一个节点,最后一个节点的后继是第一个节点。双向链表: 在构建相邻节点的引用关系时,设前驱节点 pre 和当前节点 cur ,不仅应构建 pre.right = cur ,也应构建 cur.left = pre。当 pre 不为空时: 修改双向节点引用,即 pre.right = cur , cur.left = pre;构建循环链表: 中序遍历完成后,head 指向头节点, pre 指向尾节点,因此修改 head 和 pre 的双向节点引用即可;原创 2023-02-11 14:59:45 · 307 阅读 · 0 评论 -
剑指 Offer 37. 序列化二叉树
请实现两个函数,分别用来序列化和反序列化二叉树。你需要设计一个算法来实现二叉树的序列化与反序列化。这里不限定你的序列 / 反序列化算法执行逻辑,你只需要保证一个二叉树可以被序列化为一个字符串并且将这个字符串反序列化为原始的树结构。原创 2023-02-21 22:34:32 · 299 阅读 · 0 评论 -
剑指 Offer 38. 字符串的排列
输入一个字符串,打印出该字符串中字符的所有排列。你可以以任意顺序返回这个字符串数组,但里面不能有重复元素。原创 2023-03-05 23:08:36 · 85 阅读 · 0 评论 -
剑指Offer 39 数组中出现次数超过一半的数字
剑指Offer39 数组中出现次数超过一半的数字本题常见的三种解法:我们知道出现次数最多的元素大于 n/2 次,所以可以用哈希表来快速统计每个元素出现的次数。我们使用哈希映射(HashMap)来存储每个元素以及出现的次数。对于哈希映射中的每个键值对,键表示一个元素,值表示该元素出现的次数。我们用一个循环遍历数组 nums 并将数组中的每个元素加入哈希映射中。在这之后,我们遍历哈希映射中的所有键值对,返回值最大的键。我们同样也可以在遍历数组 nums 时候使用打擂台的方法,维护最大的值,这样省去了最后对哈希原创 2022-12-07 18:47:46 · 384 阅读 · 0 评论 -
剑指 Offer 40. 最小的k个数
剑指 Offer 42. 连续子数组的最大和假设nums数组的长度是n,下标从0到n−1。我们用 f(i)代表以第i个数结尾的连续子数组的最大和,那么很显然我们要求的答案就是:因此我们只需要求出每个位置的 f(i),然后返回 f 数组中的最大值即可。那么我们如何求 f(i) 呢?我们可以考虑 nums[i] 单独成为一段还是加入 f(i−1)对应的那一段,这取决于 nums[i]和f(i−1)+nums[i] 的大小,我们希望获得一个比较大的,于是可以写出这样的动态规划转移方程:不难给出一个时间复杂度 O(原创 2022-12-07 18:45:27 · 609 阅读 · 0 评论 -
剑指 Offer 41. 数据流中的中位数
如何得到一个数据流中的中位数?如果从数据流中读出奇数个数值,那么中位数就是所有数值排序之后位于中间的数值。如果从数据流中读出偶数个数值,那么中位数就是所有数值排序之后中间两个数的平均值。原创 2023-03-07 20:49:50 · 192 阅读 · 0 评论 -
剑指 Offer 42. 连续子数组的最大和
剑指 Offer 42. 连续子数组的最大和假设nums数组的长度是n,下标从0到n−1。我们用 f(i)代表以第i个数结尾的连续子数组的最大和,那么很显然我们要求的答案就是:因此我们只需要求出每个位置的 f(i),然后返回 f 数组中的最大值即可。那么我们如何求 f(i) 呢?我们可以考虑 nums[i] 单独成为一段还是加入 f(i−1)对应的那一段,这取决于 nums[i]和f(i−1)+nums[i] 的大小,我们希望获得一个比较大的,于是可以写出这样的动态规划转移方程:不难给出一个时间复杂度 O(原创 2022-12-07 18:43:39 · 181 阅读 · 0 评论