From abecfd54a584a1bb64d6cbdf47cf4057e5b32886 Mon Sep 17 00:00:00 2001 From: CyC2018 <1029579233@qq.com> Date: Sat, 17 Mar 2018 17:24:38 +0800 Subject: [PATCH] auto commit --- notes/剑指 offer 题解.md | 184 ++++++++++++++++++++------------------- 1 file changed, 93 insertions(+), 91 deletions(-) diff --git a/notes/剑指 offer 题解.md b/notes/剑指 offer 题解.md index 2da38375..472b720d 100644 --- a/notes/剑指 offer 题解.md +++ b/notes/剑指 offer 题解.md @@ -1,5 +1,7 @@ * [前言](#前言) + * [变量命名规范](#变量命名规范) + * [复杂度简写说明](#复杂度简写说明) * [面试需要的基础知识](#面试需要的基础知识) * [2. 实现 Singleton](#2-实现-singleton) * [3. 数组中重复的数字](#3-数组中重复的数字) @@ -88,7 +90,7 @@ # 前言 -**变量命名规范** +## 变量命名规范 - nums 表示数组,matrix 表示矩阵; - n 表示数组长度、字符串长度、树节点个数,以及其它具有一维性质的数据结构的元素个数; @@ -99,7 +101,7 @@ - ret 表示结果相关的变量; - dp 表示动态规划保存子问题的数组; -**复杂度简写说明** +## 复杂度简写说明 O(nlogn) + O(n2),第一个指时间复杂度,第二个指空间复杂度。 @@ -111,11 +113,11 @@ O(nlogn) + O(n2),第一个指时间复杂度,第二个 ## 3. 数组中重复的数字 -**题目描述** +### 题目描述 在一个长度为 n 的数组里的所有数字都在 0 到 n-1 的范围内。数组中某些数字是重复的,但不知道有几个数字是重复的。也不知道每个数字重复几次。请找出数组中任意一个重复的数字。例如,如果输入长度为 7 的数组 {2, 3, 1, 0, 2, 5, 3},那么对应的输出是第一个重复的数字 2。 -**解题思路** +### 解题思路 这种数组元素在 [0, n-1] 范围内的问题,可以将值为 i 的元素放到第 i 个位置上。 @@ -156,7 +158,7 @@ private void swap(int[] nums, int i, int j) { ## 4. 二维数组中的查找 -**题目描述** +### 题目描述 在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。 @@ -174,7 +176,7 @@ Given target = 5, return true. Given target = 20, return false. ``` -**解题思路** +### 解题思路 从右上角开始查找。因为矩阵中的一个数,它左边的数都比它来的小,下边的数都比它来的大。因此,从右上角开始查找,就可以根据 target 和当前元素的大小关系来改变行和列的下标,从而缩小查找区间。 @@ -196,11 +198,11 @@ public boolean Find(int target, int[][] matrix) { ## 5. 替换空格 -**题目描述** +### 题目描述 请实现一个函数,将一个字符串中的空格替换成“%20”。例如,当字符串为 We Are Happy. 则经过替换之后的字符串为 We%20Are%20Happy。 -**解题思路** +### 解题思路 现在字符串尾部填充任意字符,使得字符串的长度等于将字符串替换之后的长度。因为一个空格要替换成三个字符(%20),因此当遍历到一个空格时,需要在尾部填充两个任意字符。 @@ -238,11 +240,11 @@ public String replaceSpace(StringBuffer str) { ## 6. 从尾到头打印链表 -**题目描述** +### 题目描述 输入链表的第一个节点,从尾到头反过来打印出每个结点的值。 -**解题思路** +### 解题思路 栈 @@ -315,7 +317,7 @@ public ArrayList printListFromTailToHead(ListNode listNode) { ## 7. 重建二叉树 -**题目描述** +### 题目描述 根据二叉树的前序遍历和中序遍历的结果,重建出该二叉树。 @@ -334,7 +336,7 @@ Return the following binary tree: 15 7 ``` -**解题思路** +### 解题思路 前序遍历的第一个值为树根节点的值,使用这个值将中序遍历结果分成两部分,左部分为树的左子树中序遍历结果,右部分为树的右子树中序遍历的结果。 @@ -362,11 +364,11 @@ private TreeNode reConstructBinaryTree(int[] pre, int preL, int preR, int[] in, ## 8. 二叉树的下一个结点 -**题目描述** +### 题目描述 给定一个二叉树和其中的一个结点,请找出中序遍历顺序的下一个结点并且返回。注意,树中的结点不仅包含左右子结点,同时包含指向父结点的指针。 -**解题思路** +### 解题思路 - 如果一个节点有右子树不为空,那么该节点的下一个节点是右子树的最左节点; @@ -408,7 +410,7 @@ public TreeLinkNode GetNext(TreeLinkNode pNode) { ## 9. 用两个栈实现队列 -**解题思路** +### 解题思路 in 栈用来处理入栈(push)操作,out 栈用来处理出栈(pop)操作。一个元素进入 in 栈之后,出栈的顺序被反转。当元素要出栈时,需要先进入 out 栈,此时元素出栈顺序再一次被反转,因此出栈顺序就和最开始入栈顺序是相同的,此时先进入的元素先退出,这就是队列的顺序。 @@ -432,13 +434,13 @@ public int pop() { ## 10.1 斐波那契数列 -**题目描述** +### 题目描述 以 O(1) 的时间复杂度求菲波那切数列。

-**解题思路** +### 解题思路 如果使用递归求解,那么会重复计算一些子问题。例如,求 f(10) 需要计算 f(9) 和 f(8),计算 f(9) 需要计算 f(8) 和 f(7),可以看到 f(8) 被重复计算了。 @@ -464,7 +466,7 @@ public class Solution { ## 10.2 跳台阶 -**题目描述** +### 题目描述 一只青蛙一次可以跳上 1 级台阶,也可以跳上 2 级。求该青蛙跳上一个 n 级的台阶总共有多少种跳法。 @@ -483,7 +485,7 @@ public int JumpFloor(int n) { ## 10.3 变态跳台阶 -**题目描述** +### 题目描述 一只青蛙一次可以跳上 1 级台阶,也可以跳上 2 级……它也可以跳上 n 级。求该青蛙跳上一个 n 级的台阶总共有多少种跳法。 @@ -502,7 +504,7 @@ public int JumpFloorII(int n) { ## 10.4 矩形覆盖 -**题目描述** +### 题目描述 我们可以用 2\*1 的小矩形横着或者竖着去覆盖更大的矩形。请问用 n 个 2\*1 的小矩形无重叠地覆盖一个 2\*n 的大矩形,总共有多少种方法? @@ -521,11 +523,11 @@ public int RectCover(int n) { ## 11. 旋转数组的最小数字 -**题目描述** +### 题目描述 把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。输入一个非递减排序的数组的一个旋转,输出旋转数组的最小元素。例如数组 {3, 4, 5, 1, 2} 为 {1, 2, 3, 4, 5} 的一个旋转,该数组的最小值为 1。NOTE:给出的所有元素都大于 0,若数组大小为 0,请返回 0。 -**解题思路** +### 解题思路 分治 @@ -565,7 +567,7 @@ public int minNumberInRotateArray(int[] array) { ## 12. 矩阵中的路径 -**题目描述** +### 题目描述 请设计一个函数,用来判断在一个矩阵中是否存在一条包含某字符串所有字符的路径。路径可以从矩阵中的任意一个格子开始,每一步可以在矩阵中向左,向右,向上,向下移动一个格子。如果一条路径经过了矩阵中的某一个格子,则该路径不能再进入该格子。例如 a b c e s f c s a d e e 矩阵中包含一条字符串 "bcced" 的路径,但是矩阵中不包含 "abcb" 路径,因为字符串的第一个字符 b 占据了矩阵中的第一行第二个格子之后,路径不能再次进入该格子。 @@ -610,7 +612,7 @@ private boolean backtracking(char[][] matrix, char[] str, boolean[][] used, int ## 13. 机器人的运动范围 -**题目描述** +### 题目描述 地上有一个 m 行和 n 列的方格。一个机器人从坐标 (0, 0) 的格子开始移动,每一次只能向左右上下四个方向移动一格,但是不能进入行坐标和列坐标的数位之和大于 k 的格子。例如,当 k 为 18 时,机器人能够进入方格(35, 37),因为 3+5+3+7=18。但是,它不能进入方格(35, 38),因为 3+5+3+8=19。请问该机器人能够达到多少个格子? @@ -656,7 +658,7 @@ private void initDigitSum(int rows, int cols) { ## 14. 剪绳子 -**题目描述** +### 题目描述 把一根绳子剪成多段,并且使得每段的长度乘积最大。 @@ -682,7 +684,7 @@ int maxProductAfterCuttin(int length) { ## 15. 二进制中 1 的个数 -**题目描述** +### 题目描述 输入一个整数,输出该数二进制表示中 1 的个数。其中负数用补码表示 @@ -713,11 +715,11 @@ public int NumberOf1(int n) { ## 16. 数值的整数次方 -**题目描述** +### 题目描述 给定一个 double 类型的浮点数 base 和 int 类型的整数 exponent。求 base 的 exponent 次方。 -**解题思路** +### 解题思路 下面的讨论中 x 代表 base,N 代表 exponent。 @@ -743,11 +745,11 @@ public double Power(double base, int exponent) { ## 17. 打印从 1 到最大的 n 位数 -**题目描述** +### 题目描述 输入数字 n,按顺序打印出从 1 最大的 n 位十进制数。比如输入 3,则打印出 1、2、3 一直到最大的 3 位数即 999。 -**解题思路** +### 解题思路 由于 n 可能会非常大,因此不能直接用 int 存储数,而是用 char 数组进行存储。 @@ -783,7 +785,7 @@ private void printNumber(char[] number) { ## 18.1 在 O(1) 时间内删除链表节点 -**解题思路** +### 解题思路 - 如果链表不是尾节点,那么可以直接将下一个节点的值赋给节点,令节点指向下下个节点,然后删除下一个节点,时间复杂度为 O(1)。 @@ -815,7 +817,7 @@ public ListNode deleteNode(ListNode head, ListNode tobeDelete) { ## 18.2 删除链表中重复的结点 -**题目描述** +### 题目描述 ```html Input : 1->2->3->3->4->4->5 @@ -845,7 +847,7 @@ public ListNode deleteDuplication(ListNode pHead) { ## 19. 正则表达式匹配 -**题目描述** +### 题目描述 请实现一个函数用来匹配包括 '.' 和 '\*' 的正则表达式。模式中的字符 '.' 表示任意一个字符,而 '\*' 表示它前面的字符可以出现任意次(包含 0 次)。 在本题中,匹配是指字符串的所有字符匹配整个模式。例如,字符串 "aaa" 与模式 "a.a" 和 "ab\*ac\*a" 匹配,但是与 "aa.a" 和 "ab\*a" 均不匹配. @@ -878,7 +880,7 @@ public boolean match(char[] s, char[] p) { ## 20. 表示数值的字符串 -**题目描述** +### 题目描述 请实现一个函数用来判断字符串是否表示数值(包括整数和小数)。例如,字符串 "+100","5e2","-123","3.1416" 和 "-1E-16" 都表示数值。 但是 "12e","1a3.14","1.2.3","+-5" 和 "12e+4.3" 都不是。 @@ -895,7 +897,7 @@ public boolean isNumeric(char[] str) { 保证奇数和奇数,偶数和偶数之间的相对位置不变,这和书本不太一样。 -**解题思路** +### 解题思路 - 时间复杂度 : O(n2) - 空间复杂度 : O(1) @@ -936,7 +938,7 @@ public void reOrderArray(int[] array) { ## 22. 链表中倒数第 K 个结点 -**解题思路** +### 解题思路 设链表的长度为 N。设两个指针 P1 和 P2,先让 P1 移动 K 个节点,则还有 N - K 个节点可以移动。此时让 P1 和 P2 同时移动,可以知道当 P1 移动到链表结尾时,P2 移动到 N - K 个节点处,该位置就是倒数第 K 个节点。 @@ -959,7 +961,7 @@ public ListNode FindKthToTail(ListNode head, int k) { ## 23. 链表中环的入口结点 -**解题思路** +### 解题思路 使用双指针,一个指针 fast 每次移动两个节点,一个指针 slow 每次移动一个节点。因为存在环,所以两个指针必定相遇在环中的某个节点上。此时 fast 移动的节点数为 x+2y+z,slow 为 x+y,由于 fast 速度比 slow 快一倍,因此 x+2y+z=2(x+y),得到 x=z。 @@ -989,7 +991,7 @@ public ListNode EntryNodeOfLoop(ListNode pHead) { ## 24. 反转链表 -**解题思路** +### 解题思路 头插法 @@ -1008,7 +1010,7 @@ public ListNode ReverseList(ListNode head) { ## 25. 合并两个排序的链表 -**题目描述** +### 题目描述

@@ -1034,7 +1036,7 @@ public ListNode Merge(ListNode list1, ListNode list2) { ## 26. 树的子结构 -**题目描述** +### 题目描述

@@ -1057,7 +1059,7 @@ private boolean isSubtree(TreeNode root1, TreeNode root2) { ## 27. 二叉树的镜像 -**题目描述** +### 题目描述

@@ -1074,7 +1076,7 @@ public void Mirror(TreeNode root) { ## 28.1 对称的二叉树 -**题目描述** +### 题目描述 ```html 1 @@ -1100,7 +1102,7 @@ boolean isSymmetrical(TreeNode t1, TreeNode t2) { ## 28.2 平衡二叉树 -**题目描述** +### 题目描述 ```html 3 @@ -1131,7 +1133,7 @@ private int height(TreeNode root) { ## 29. 顺时针打印矩阵 -**题目描述** +### 题目描述 下图的矩阵顺时针打印结果为:1, 2, 3, 4, 8, 12, 16, 15, 14, 13, 9, 5, 6, 7, 11, 10 @@ -1154,7 +1156,7 @@ public ArrayList printMatrix(int[][] matrix) { ## 30. 包含 min 函数的栈 -**题目描述** +### 题目描述 定义栈的数据结构,请在该类型中实现一个能够得到栈最小元素的 min 函数。 @@ -1186,7 +1188,7 @@ public int min() { ## 31. 栈的压入、弹出序列 -**题目描述** +### 题目描述 输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否为该栈的弹出顺序。假设压入栈的所有数字均不相等。例如序列 1,2,3,4,5 是某栈的压入顺序,序列 4,5,3,2,1 是该压栈序列对应的一个弹出序列,但 4,3,5,1,2 就不可能是该压栈序列的弹出序列。 @@ -1207,7 +1209,7 @@ public boolean IsPopOrder(int[] pushA, int[] popA) { ## 32.1 从上往下打印二叉树 -**题目描述** +### 题目描述 从上往下打印出二叉树的每个节点,同层节点从左至右打印。 @@ -1215,7 +1217,7 @@ public boolean IsPopOrder(int[] pushA, int[] popA) {

-**解题思路** +### 解题思路 使用队列来进行层次遍历。 @@ -1242,7 +1244,7 @@ public ArrayList PrintFromTopToBottom(TreeNode root) { ## 32.2 把二叉树打印成多行 -**题目描述** +### 题目描述 和上题几乎一样。 @@ -1269,7 +1271,7 @@ ArrayList> Print(TreeNode pRoot) { ## 32.3 按之字形顺序打印二叉树 -**题目描述** +### 题目描述 请实现一个函数按照之字形打印二叉树,即第一行按照从左到右的顺序打印,第二层按照从右至左的顺序打印,第三行按照从左到右的顺序打印,其他行以此类推。 @@ -1303,7 +1305,7 @@ public ArrayList> Print(TreeNode pRoot) { ## 33. 二叉搜索树的后序遍历序列 -**题目描述** +### 题目描述 输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历的结果。 @@ -1331,7 +1333,7 @@ private boolean verify(int[] sequence, int first, int last) { ## 34. 二叉树中和为某一值的路径 -**题目描述** +### 题目描述 输入一颗二叉树和一个整数,打印出二叉树中结点值的和为输入整数的所有路径。路径定义为从树的根结点开始往下一直到叶结点所经过的结点形成一条路径。 @@ -1363,13 +1365,13 @@ private void dfs(TreeNode node, int target, int curSum, ArrayList path) ## 35. 复杂链表的复制 -**题目描述** +### 题目描述 输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的 head。(注意,输出结果中请不要返回参数中的节点引用,否则判题程序会直接返回空)

-**解题思路** +### 解题思路 第一步,在每个节点的后面插入复制的节点。 @@ -1419,7 +1421,7 @@ public RandomListNode Clone(RandomListNode pHead) { ## 36. 二叉搜索树与双向链表 -**题目描述** +### 题目描述 输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表。要求不能创建任何新的结点,只能调整树中结点指针的指向。 @@ -1446,7 +1448,7 @@ private void inOrder(TreeNode node) { ## 37. 序列化二叉树 -**题目描述** +### 题目描述 请实现两个函数,分别用来序列化和反序列化二叉树。 @@ -1482,7 +1484,7 @@ public class Solution { ## 38. 字符串的排列 -**题目描述** +### 题目描述 输入一个字符串 , 按字典序打印出该字符串中字符的所有排列。例如输入字符串 abc, 则打印出由字符 a, b, c 所能排列出来的所有字符串 abc, acb, bac, bca, cab 和 cba。 @@ -1518,7 +1520,7 @@ private void backtracking(char[] chars, boolean[] hasUsed, StringBuffer s) { ## 39. 数组中出现次数超过一半的数字 -**解题思路** +### 解题思路 多数投票问题,可以利用 Boyer-Moore Majority Vote Algorithm 来解决这个问题,使得时间复杂度为 O(n)。 @@ -1635,7 +1637,7 @@ public ArrayList GetLeastNumbers_Solution(int[] input, int k) { ## 41.1 数据流中的中位数 -**题目描述** +### 题目描述 如何得到一个数据流中的中位数?如果从数据流中读出奇数个数值,那么中位数就是所有数值排序之后位于中间的数值。如果从数据流中读出偶数个数值,那么中位数就是所有数值排序之后中间两个数的平均值。 @@ -1675,7 +1677,7 @@ public class Solution { ## 41.2 字符流中第一个不重复的字符 -**题目描述** +### 题目描述 请实现一个函数用来找出字符流中第一个只出现一次的字符。例如,当从字符流中只读出前两个字符 "go" 时,第一个只出现一次的字符是 "g"。当从该字符流中读出前六个字符“google" 时,第一个只出现一次的字符是 "l"。 @@ -1701,7 +1703,7 @@ public class Solution { ## 42. 连续子数组的最大和 -**题目描述** +### 题目描述 {6,-3,-2,7,-15,1,2,2},连续子向量的最大和为 8(从第 0 个开始,到第 3 个为止) @@ -1736,7 +1738,7 @@ public int NumberOf1Between1AndN_Solution(int n) { ## 44. 数字序列中的某一位数字 -**题目描述** +### 题目描述 数字以 0123456789101112131415... 的格式序列化到一个字符串中,求这个字符串的第 index 位。 @@ -1785,11 +1787,11 @@ private int beginNumber(int digit) { ## 45. 把数组排成最小的数 -**题目描述** +### 题目描述 输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个。例如输入数组 {3,32,321},则打印出这三个数字能排成的最小数字为 321323。 -**解题思路** +### 解题思路 可以看成是一个排序问题,在比较两个字符串 S1 和 S2 的大小时,应该比较的是 S1+S2 和 S2+S1 的大小,如果 S1+S2 >= S2+S1,那么应该把 S1 排在前面,否则应该把 S2 排在前面。 @@ -1807,7 +1809,7 @@ public String PrintMinNumber(int[] numbers) { ## 46. 把数字翻译成字符串 -**题目描述** +### 题目描述 给定一个数字,按照如下规则翻译成字符串:0 翻译成“a”,1 翻译成“b”...25 翻译成“z”。一个数字有多种翻译可能,例如 12258 一共有 5 种,分别是 bccfi,bwfi,bczi,mcfi,mzi。实现一个函数,用来计算一个数字有多少种不同的翻译方法。 @@ -1829,7 +1831,7 @@ public int getTranslationCount(String number) { ## 47. 礼物的最大价值 -**题目描述** +### 题目描述 在一个 m\*n 的棋盘的每一个格都放有一个礼物,每个礼物都有一定价值(大于 0)。从左上角开始拿礼物,每次向右或向下移动一格,直到右下角结束。给定一个棋盘,求拿到礼物的最大价值。例如,对于如下棋盘 @@ -1842,7 +1844,7 @@ public int getTranslationCount(String number) { 礼物的最大价值为 1+12+5+7+7+16+5=53。 -**解题思路** +### 解题思路 应该用动态规划求解,而不是深度优先搜索,深度优先搜索过于复杂,不是最优解。 @@ -1864,7 +1866,7 @@ public int getMaxValue(int[][] values) { ## 48. 最长不含重复字符的子字符串 -**题目描述** +### 题目描述 输入一个字符串(只包含 a\~z 的字符),求其最长不含重复字符的子字符串的长度。例如对于 arabcacfr,最长不含重复字符的子字符串为 acfr,长度为 4。 @@ -1890,7 +1892,7 @@ public int longestSubStringWithoutDuplication(String str) { ## 49. 丑数 -**题目描述** +### 题目描述 把只包含因子 2、3 和 5 的数称作丑数(Ugly Number)。例如 6、8 都是丑数,但 14 不是,因为它包含因子 7。 习惯上我们把 1 当做是第一个丑数。求按从小到大的顺序的第 N 个丑数。 @@ -1926,7 +1928,7 @@ public int FirstNotRepeatingChar(String str) { ## 51. 数组中的逆序对 -**题目描述** +### 题目描述 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数 P。 @@ -1968,7 +1970,7 @@ private void merge(int[] nums, int start, int mid, int end) { ## 52. 两个链表的第一个公共结点 -**题目描述** +### 题目描述 ```html A: a1 → a2 @@ -1978,7 +1980,7 @@ A: a1 → a2 B: b1 → b2 → b3 ``` -**解题思路** +### 解题思路 设 A 的长度为 a + c,B 的长度为 b + c,其中 c 为尾部公共部分长度,可知 a + c + b = b + c + a。 @@ -1999,7 +2001,7 @@ public ListNode FindFirstCommonNode(ListNode pHead1, ListNode pHead2) { ## 53 数字在排序数组中出现的次数 -**题目描述** +### 题目描述 ```html Input: @@ -2009,7 +2011,7 @@ Output: 4 ```` -**解题思路** +### 解题思路 可以用二分查找找出数字在数组的最左端和最右端。 @@ -2045,7 +2047,7 @@ private int getLastK(int[] nums, int K) { ## 54. 二叉搜索树的第 k 个结点 -**解题思路** +### 解题思路 利用二叉搜索数中序遍历有序的特点。 @@ -2078,11 +2080,11 @@ public int TreeDepth(TreeNode root) { ## 56. 数组中只出现一次的数字 -**题目描述** +### 题目描述 一个整型数组里除了两个数字之外,其他的数字都出现了两次,找出这两个数。 -**解题思路** +### 解题思路 两个不相等的元素在位级表示上必定会有一位存在不同。 @@ -2105,11 +2107,11 @@ public void FindNumsAppearOnce(int[] array, int num1[], int num2[]) { ## 57.1 和为 S 的两个数字 -**题目描述** +### 题目描述 输入一个递增排序的数组和一个数字 S,在数组中查找两个数,是的他们的和正好是 S,如果有多对数字的和等于 S,输出两个数的乘积最小的。 -**解题思路** +### 解题思路 使用双指针,一个指针指向元素较小的值,一个指针指向元素较大的值。指向较小元素的指针从头向尾遍历,指向较大元素的指针从尾向头遍历。 @@ -2130,7 +2132,7 @@ public ArrayList FindNumbersWithSum(int[] array, int sum) { ## 57.2 和为 S 的连续正数序列 -**题目描述** +### 题目描述 和为 100 的连续序列有 18, 19, 20, 21, 22 @@ -2165,13 +2167,13 @@ public ArrayList> FindContinuousSequence(int sum) { ## 58.1 翻转单词顺序列 -**题目描述** +### 题目描述 输入:"I am a student." 输出:"student. a am I" -**解题思路** +### 解题思路 题目应该有一个隐含条件,就是不能用额外的空间。虽然 Java 的题目输入参数为 String 类型,需要先创建一个字符数组使得空间复杂度为 O(n),但是正确的参数类型应该和原书一样,为字符数组,并且只能使用该字符数组的空间。任何使用了额外空间的解法在面试时都会大打折扣,包括递归解法。正确的解法应该是和书上一样,先旋转每个单词,再旋转整个字符串。 @@ -2205,7 +2207,7 @@ private void reverse(char[] c, int start, int end) { ## 58.2 左旋转字符串 -**题目描述** +### 题目描述 对于一个给定的字符序列 S,请你把其循环左移 K 位后的序列输出。例如,字符序列 S=”abcXYZdef”, 要求输出循环左移 3 位后的结果,即“XYZdefabc”。 @@ -2232,7 +2234,7 @@ private void reverse(char[] c, int i, int j) { ## 59. 滑动窗口的最大值 -**题目描述** +### 题目描述 给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值。例如,如果输入数组 {2, 3, 4, 2, 6, 2, 5, 1} 及滑动窗口的大小 3,那么一共存在 6 个滑动窗口,他们的最大值分别为 {4, 4, 6, 6, 6, 5}; @@ -2255,7 +2257,7 @@ public ArrayList maxInWindows(int[] num, int size) { ## 60. n 个骰子的点数 -**题目描述** +### 题目描述 把 n 个骰子仍在地上,求点数和为 s 的概率。 @@ -2315,7 +2317,7 @@ public double countProbability(int n, int s) { ## 61. 扑克牌顺子 -**题目描述** +### 题目描述 五张牌,其中大小鬼为癞子,牌面大小为 0。判断是否能组成顺子。 @@ -2337,11 +2339,11 @@ public boolean isContinuous(int [] numbers) { ## 62. 圆圈中最后剩下的数 -**题目描述** +### 题目描述 让小朋友们围成一个大圈。然后 , 他随机指定一个数 m, 让编号为 0 的小朋友开始报数。每次喊到 m-1 的那个小朋友要出列唱首歌 , 然后可以在礼品箱中任意的挑选礼物 , 并且不再回到圈中 , 从他的下一个小朋友开始 , 继续 0...m-1 报数 .... 这样下去 .... 直到剩下最后一个小朋友 , 可以不用表演。 -**解题思路** +### 解题思路 约瑟夫环 @@ -2355,7 +2357,7 @@ public int LastRemaining_Solution(int n, int m) { ## 63. 股票的最大利润 -**题目描述** +### 题目描述 可以有一次买入和一次卖出,买入必须在前。求最大收益。 @@ -2375,7 +2377,7 @@ public int maxProfit(int[] prices) { ## 64. 求 1+2+3+...+n -**题目描述** +### 题目描述 求 1+2+3+...+n,要求不能使用乘除法、for、while、if、else、switch、case 等关键字及条件判断语句(A?B:C) @@ -2400,7 +2402,7 @@ public int Add(int num1, int num2) { ## 66. 构建乘积数组 -**题目描述** +### 题目描述 给定一个数组 A[0, 1,..., n-1], 请构建一个数组 B[0, 1,..., n-1], 其中 B 中的元素 B[i]=A[0]\*A[1]\*...\*A[i-1]\*A[i+1]\*...\*A[n-1]。不能使用除法。