类别1 数组(10)
1. 二维数组中的查找
(核心技巧:从一个角找起,每次放弃一行或一列)
if...else这种还是写括号的好,像这种
if(target>array[i][j]){
j++;
}else{
i--;
}
不要简写,也不要写两个if,不然可能出错
13. 调整数组顺序使奇数位于偶数前面 (插排)
(核心技巧:插排)
两种解法:解法1:空间替代时间 解法2:插排的思想
遇到一道难题,当想不出来特别好的解法,或者想到一个思路,不知道怎么做时,可以想想暴力解法,有没有可以解决的,最起码比解不出来强。变量名一定不要写错,题目给出的变量或者前面自己定义的变量,一定要写对。array=arr,这样写不行,要把arr里的数遍历一遍,赋值给array。
28. 数组中出现次数超过一半的数字
(核心技巧:消除法及其应用)
两种解法:解法1:利用库函数排序,然后找到中间的数,然后判断它是不是出现次数超过一半的数。O(nlogn)。解法2:利用消除法找到可能的目标数字,利用times记录数字出现的次数,result记录数字,如果数组中不存在目标数字,那么无所谓,反正最后还要判断,如果存在目标数字,那么result最终的结果一定是目标数字,再判断确认一下即可。0(n)
32. 把数组排成最小的数
核心技巧(利用数组排序;比较器的使用;比较方法的正确选择)
暴力解法,O(n!)还不会写
懂这个技巧就会,不懂就不会。当然,里面涉及到的知识点,如String.valueOf(),Comparator,Arrays.sort(),compare,compareTo等也得会。
Comparator在java.utils包中,覆写的方法应该是public(原因),sort()方法结束后应该写封号。
字符串之间不能用> <比较,要用compareTo()方法
思路:新建一个字符串数组,把整数数组中的数字转化为字符串(String.valueOf(int))。然后对这个数组进行排序,规则(若str1+str2>str2+str1,则str1>str2)。最后,拼接整个数组的字符串为一个字符串,即为所求。
String -->
Integer.parseInt(str)
Integer.valueOf(str).intValue()
50. 数组中重复的数字(可以用hashMap)
HashMap可以解决,但空间复杂度会大。
一个技巧,数字在0-n-1之间,可以监控数字为下标位置上的数,出现第一次时,给它加length,第二次出现时就可以知道重复出现过了。注意数组改变过了,需要取余操作。
51. 构建乘积数组
可以用除法,是最简单的,一个for循环就够了,但是不让用;双重循环也比较好做,但时间复杂度高;最好的方法是上三角、下三角那样做,纯技巧,记得就会,不记得就不会。
leetcode1 两数之和
hashmap很好的使用例子,可以再o(n)时间内解决,API,hashmap.containsKey(()
leetcode11 盛最多水的容器
笔试题应该不会考,暴力解o(n^2), 双指针,i, j,但没想到怎么用双指针,技巧是左边低,i++,右边低,j--
leetcode31 下一个排列
笔试可能会考
思想:一个数组,从右向左,如果一直是升序的,就是字典序最大的,所以要找到第一个i和i+1,使得nums[i] 注意点,reverse(nums,i),交换的是j和nums.lenth+i-1-j,条件是2*j 注意找到的是i和i+1,使得nums[i] leetcode 搜索选择排序数组: 笔试考的概率不大,因为o(n)就找完了 思想:二分查找,start, end, mid, 如target == nums[mid] return mid ,否则如果start,mid有序,在start,mid-1找,否则在mid+1,end找,如果mid+1到end有序,在mid+1,end找,,否则在start,,mid-1找,mid左右总有一个是有序的,所以总能无遗漏的找完。 leetcode48 旋转图像 将题目转化为转置+行逆排 letcode55 跳跃游戏 定义一个变量max,初始化为0,代表当前所能达到的最大位置,从0开始,如果一个位置能够到达,就尝试更新max,直到max>length-1,或者遍历到头或某位置不可到达。 leetcode56 合并区间 定义一个数组res,遍历intervals,如果遍历到的arr中,arr[0]>res[i][1],则res中新加一个,否则尝试更新res中最右一侧数组的边界。 类别2 字符串(9) 2. 替换空格 StringBuffer API及StringBuffer与String相互转化。 27. 字符串的排列 String与字符数组相互转化。看位置,每个位置上,每个元素都出现过一次,是完整的全排列。递归。 leetcode17 电话号码的字母组合 9个数字,每个数字各代表三个字母 第一步:定义HashMap,键为数字,值为对应的字符串 第二步:定义结果集,等待添加结果 第三步:定义辅助函数helper(int digital, int result),如果digital长度为0,则说明result已经从每个数字代表的字符串中拿到了一个字母,可以返回结果了;否则,利用digital.substring(0)从HashMap中拿到字符串,遍历字符串,每次执行helper(int digital.substring(1), int result)。 第四步:在给出的函数中调用helper(digital, "");,返回第二步的结果集。 34. 第一个只出现一次的字符位置 思路:用hashMap或者数组(65-90,97-122)等记录记录下,字符串中元素出现的次数。后续有两种方法,可以获得第一个出现一次的方法:1)遍历所有只出现一次的元素,看谁排在前面(indexOf(),charAt())2)从头至尾遍历字符串元素,看哪个 43. 左旋转字符串 三种思路:1)数组 2)三次反转 3)StringBuilder,注意StringBuilder是lang包中的,lang包不需要导入,substring(a,b)(不包含b),append(StringBuilder),构造函数可以使用String做参数,StringBuilder sb,sb.length()。 44. 翻转单词顺序 思路:字符串利用split(" ")转化为字符串数组,然后用一个循环(i=arr.length-1;i>0;i--),转成新的字符串。str.trim()去点字符串两端的空格。 49. 把字符串转换成整数 long转int,要强转,int转long不需要,int转char要强转,char转int不需要,但是’9‘到9,要减去’0‘ 52. 正则表达式匹配 递归判断 1)字符串和模式串都到头,成功 2)字符串没到头,模式串到头,失败 3)如果模式串下一个字符是*, 那么如果字符串和模式串当前匹配,那么有三种情况,模式串匹配当前字符,模式串匹配0个字符,模式串匹配当前字符后,继续匹配下一个字符。如果字符串和模式串不匹配,模式串只匹配当前字符。 如果模式串下一个字符不是*,有两种情况,模式串匹配当前字符,都进行下一个;如果不匹配,直接失败。 细节:.*可以是无数点的意思,可以匹配任意字符串。 如果模式串下一个字符是*,而字符串已经越界,是不匹配的,即‘’是不匹配'.'这样可以说的通,如果匹配,matchHelper(str,pattern,strIndex+1,patIndex)会一直走下去,而且其他的条件也可能越界,当然也有可能可以匹配,只是我对题目找到没理解透彻,没找到好方法。 3. 表示数值的字符串 1)java中的&& || 2)为什么我的事错的?????????????????? 54. 字符流中第一个不重复的字符 1) int转char怎么转,转过去值是怎么变的 2)hashMap的用法,不只是能记录是否是1,还能记录哪个最先出现,这样可以降低空间复杂度 3)java中char到底占几个字节,char的范围是多少,汉子怎么算?????????? leetcode 121 买卖股票的最佳时机 记录一个历史最小值(不是全局最小值),然后用当前值减历史最小值,其实也不难,但为什么想不到呢? leetcode 3 无重复字符的最长子串 基本上不会作为笔试题,因为暴力方法也很快,技巧是i,j,随着i递增,j也在递增,j不需要从i开始。 leetcode5 最长回文子串 笔试基本不会考,暴力解,O(n^3),比较好写。找到2n-1个中心点,不断寻找回文子串,记录最大值。 KMP算法 KMP算法 匹配的时候,当不匹配发生时,重点是指向主字符串的i指针不变,而模式串根据next数组向前移动。 求next数组时,关键是知道next[i-1]后,求next[i]时,可以根据p[i]与p[next[i]]相等来进行计算,减少计算过程。 类别3 链表(8) 3. 从头到尾打印链表 从头至尾放到arrayList里,用Collections.reverse()逆置一下。 14. 链表中倒数第k个结点 可以先遍历一遍,记录下结点个数,然后再计算倒数第k个结点,也可以找两个结点,p先跑k-1步,q再跑,这样q就只跑了n-k+1步,正好是倒数第k个结点的位置。 15. 反转链表 设置两个结点p,q,一前一后,就可以了 16. 合并两个排序的链表 归并排序的一部分,比较简单 25. 复杂链表的复制 三步走策略,具体见牛客网,记住了就会,记不住就O(N2) 36. 两个链表的第一个公共结点 两个链表呈Y字形 46. 圆圈中最后剩下的数(抽线建模能力) 用数组和HashMap是最蠢的 可以用链表(自定义节点),不难写 也可以用Linkedlist,比较好写 55. 链表中环的入口结点 1)快慢指针,快的一定可以追上慢的,且差几格,就走几步。2)同速指针,相遇在入口结点 56. 删除链表中重复的结点 1)删掉所有重复出现的结点而不是去重。2)可以加一个头节点作为辅助结点3)排序的 leetcode234 回文链表 O(n) 时间复杂度,O(1) 空间复杂度 将链表后半段反转,然后与前半段比较,看是否是回文链表。回文数组,双指针更简单一些。 leetcode2 两数相加 我第一次做法:将两个链表反转,然后转为数字,然后相加,然后再转为链表,然后反转,挺麻烦的,而且问题是很多数字超过int范围,甚至超过long的范围。 正确做法是直接将两个链表相加,产生节点和进位 leetcode148 排序链表 第一点是用快慢指针找到中间节点的方法;第二点是归并排序对链表进行排序的方法;第三点是merge过程利用ListNode h = new ListNode(0)简单地进行归并。 类别4 树(15) 4. 重建二叉树 1)中序,先序确定树结构的方法 2)递归 17. 树的子结构 首先递归遍历树A,看哪个结点可能是B的根节点,如果可能是,递归判断。两个递归。 18. 二叉树的镜像 递归的交换二叉树的左右结点 层次遍历,主要考察队列API Queue queue.offer(root); TreeNode node = queue.peek(); queue.poll(); 23. 二叉搜素树的后序遍历序列 二叉搜素树,又称二叉排序树,和前面几道题思路差不多,递归遍历、判断。 24. 二叉树中和为某一值的路径 1)某节点是否是根节点 if(root.left == null && root.right == null) arrayList.remove(arrayList.size()-1); 3)Comparator比较器,compare 26. 二叉搜索树与双向链表 两种方法,一种是递归方法,一种是非递归方法。递归方法可以设置一个队列,中序遍历,将结点都进队,就是有序的,然后一个个出队,这样就OK,这个其实不太好,因为这个用到了递归,用中序遍历非递归写法较好。一种是递归方法,中序遍历,先处理左子树,根节点左子树指向左子树最右结点,然后处理右子树,根节点右指针指向右子树最左节点。返回左子树最右结点。 38. 二叉树的深度 递归,两种思路,一种是简单的,返回左子树和右子树值较大的值加1,作为结果,注意递归记录递归的结果,不要再去递归一遍,否则时间复杂度会高很多。一种稍复杂,找一个全局变量,记录最大值,遇到叶子节点就判断是否更新全局变量,设置一个dpth,进到这个结点加1,遍历左右子树出这个结点时,减1。 39. 平衡二叉树 后续遍历,这样不会出现重复求子树高度的问题。高度和子树是否是平衡二叉树可以用一个变量来表示,如果子树不是平衡二叉树,那就返回-1,反则返回高度。 57. 二叉树的下一个结点 二叉树的下一个结点,有三种情况,有右子树,那就是右子树最左结点。没有右子树,有父节点的话,如果是父节点的左孩子,就是父节点。如果不是,就往上找,如果某个祖先结点是其父节点的左孩子,就是这个祖先结点的父节点。如果上述情况都不适合,就是null。 58. 对称的二叉树 对称的二叉树一定是完全二叉树,只要对称位置的两个结点值相等就可以了。所以递归判断对称位置的结点是否相同。两个结点的值要相等,左节点的左子树要和右节点的右子树相等,左节点的右子树要和右节点的左子树相等。 59. 按之字形顺序打印二叉树 两个栈,将根节点入栈1,然后在两个栈不全空的情况下循环,标志位flag为奇数,就将1中元素出栈,并将左右节点入栈2,然后将出栈的元素按要求放入返回结果中。flag为2反之,直到两个栈全空。 60. 把二叉树打印成多行 同59,且比59简单。 61 序列化二叉树 序列化二叉树:用一个StringBuffer,先序遍历,把根节点放进去,然后分别添加左子树和右子树 反序列化二叉树:先序遍历,用一个int表征遍历到的数字,因为遍历的顺序就是二叉树节点先序的顺序,所以可以通过先序遍历建树。因为是先序遍历,不会产生flag越界的情况。这个方法可以用来利用数组等建立二叉树。 62. 二叉搜索树的第k个结点 中序遍历的结果就是二叉搜索树从小到大的结果,所以用一个int flag表征遍历到的数字,flag == k时,就是结果。 leetcode437 路径总和iii 1.递归的思想,一棵树,结果是头节点+左节点+右节点; 2.一种target-val的思想,两数和的问题也是这种思想。 leetcode543 二叉树的直径 后序遍历比较好 leetcode538 把二叉搜索树转化为累加树 反中序遍历 右-中-左 leetcode617 合并二叉树 两个树的遍历,同时遍历两个树,然后分别判断是否为空,为空则返回另一个,两个都为空,就会返回空,不空相加,然后,去做左子树和右子树。 leetcode94 二叉树的中序遍历、前序遍历、后续遍历 递归写法很简单,而非递归方法有点难,前序遍历 前序遍历和中序差不多,以三个节点的树思考,区别在于前序是进栈时将其添加入结果中,而中序是出栈时添加 后序遍历,一个栈,一个LinkedList,队列用来存放结果,先将头节点放进队列里,然后逐次插入右和左,这样就是左、、右、中了。 leetcode98 验证二叉搜索树 1. helper(root, left, right)这样能保证某个节点不光大于(小于)父节点,而且也小于(大于)再上面的节点。 2.当想找一个初始的最小(最大值)时,可以不用Integer.MAX_VALUE,而把它设置为Integer类型的null,在不为null时比较,null时更新,这样就能避免真的等于MAX_VALUE的问题。 leetcode 114二叉树展开为链表 设置一个pre=null, 按后、中、头的顺序遍历,每个节点right指向pre节点,left指向null, pre=当前节点(二叉树的所有指针刚好构成一个双向链表,单向链表要舍弃一半的节点) 类别5 栈和队列(4) 5. 用两个栈实现队列 util包中数据结构大小用size() 20. 包含min函数的栈 辅助栈 21. 栈的压入、弹出序列 利用辅助栈;遍历出栈序列,如果辅助栈为空,从入栈序列入栈一个数字,判断栈顶元素和出栈序列第i个数是否相等,不等的话入栈,直到相等或者入栈序列为空。如果相等了,出栈。如果是正确的出栈序列,最后栈会为空。黑线部分可以保证一个元素出栈,其之前的元素都已经入栈。 leetcode2 有效的括号 栈使用的很好的例子,不是所有元素都要入栈的,一部分元素入栈,另一部分元素与栈顶元素判断(栈顶元素往往很敏感,看是否能消掉,最后往往会判断栈是否为空。 64. 滑动窗口的最大值 类别6 查找和排序(3) 6.查找和排序 37. 数字在排序数组中出现的次数(查找) (核心技巧:二分查找) 二分查找的应用。二分查找,递归和循环都得会。稍微改进的二分查找,可以查到数字在排序数组中第一次出现的位置,也可以查到数字在排序数组中最后一次出现的位置。 29. 最小的K个数 利用大顶堆 最大的K个数,小顶堆 leetcode75 颜色分类 荷兰国旗问题,双指针两端遍历,i,j分别代表小于i的地方为0,j代表大于j的地方为1。cur从0开始遍历,如果等于0,和i交换,i++,cur++,如果等于1,cur++,如果等于2,与j交换,cur不变,j--。 类别7 递归和循环(5) 关键1 寻找递归结束条件 ;2 寻找后面的值如何由前面的计算得到 7.斐波那契数列 斐波那契数列从第0项开始:第0项,0;第1项1,第2项1,..., 8.跳台阶 普通题目,可是在leetcode上递归无法通过,因为递归时间复杂度较高,要用循环,通过循环计算出总数目 9.变态跳台阶 主要在于n怎么从n-1或n-2等得到,这个题目是f(n)=2*f(n-1)这个还挺难像的 10. 矩形覆盖 难点与9相同,这个是斐波那契序列的规则 47. 求1+2+3+...+n 这个难点在递归结束的条件,通过逻辑短路来结束循环 类别 8 位运算(3) 11. 二进制中1的个数 数字循环右移,与1做&运算,直到结束; 40. 数组中只出现一次的数字(位运算 或hashMap) HashMap可以解决,但空间复杂度会大,HashMap api https://www.cnblogs.com/scyq/p/11667798.html 位运算解决:异或整个数组,求出两个只出现一次的数字的异或结果,写一个函数,找出异或结果右起第一个为1的位置,异或结果为1,说明这两个数字在这一位上不相同。根据这个特点,将数组中的数分为两组,分别异或,就可以求出这个数了。 注意: 48. 不用加减乘除做加法 同位相加不考虑进位,做异或运算;做&运算左移,计算进位,两者再相见,直到没有后者,前者就是结果 类别9 数据流(1) 63. 数据流中的中位数 类别10 回溯法(2) 65. 矩阵中的路径 回溯法的应用,从一个点开始,向四周发散,不断试探,直到成功,或者不成功返回false 这道题,循环遍历数组,试一下从这个点开始,是否能走通,能走通返回true,直到所有点都试一遍。需要一个数组记录这个点是否走过。 rows行,cols列的数组,以一维表示时,(i,j)在数组中下标为arr[i*cols+j] 66. 机器人的运动范围 回溯法的应用,这道题,从(0,0)开始,不断试探,是否符合条件或者是否走过,走过返回0,不符合条件返回0,否则返回1+这个节点上下左右。 leetcode39 组合总和 回溯+剪枝; leetcode46 全排列 回溯的应用,不好判断的就是这个数是否在这个结果中了,用contains可以判断,但时间复杂度较高,所以可用官方题解里的一些方法。 leetcode49 字母异位词分组 这个挺好玩,每个单词转为26个字母每个字母出现次数的字符串,作为键,值为相同键的字符串,返回new ArrayList(hashMap.values())(将java.util.HashMap$Values转为java.util.ArrayList)。 leetcode79 单词搜索 遍历二维数组每个位置,从每个位置开始,各方向尝试并回溯,注意不是到达长度才去比较,是每个位置都比较,尽快回溯。 directions, inArea, flag leetcode200 岛屿数量 方法与79相似,遍历二维数组每个位置,如果这个位置没被遍历过,且值为0,则将总数加1,否则遍历下一个;遍历的话,利用directions深度优先搜索,把联通的位置都遍历一遍。 类别11 滑动窗口(2) 41. 和为S的连续正数序列 滑动窗口,i=1,j=2,小于value,j++,大于value,i++,直到i==j 42. 和为S的两个数字 滑动窗口,从两端滑,直到碰到符合条件的 类别12 动态规划(3) 67. 剪绳子 数学方法:对3取余,根据余数不同,加不同个数的2; 动态规划:dp[i],表示i绳子为i时的最大值,返回dp[n]即可。 30. 连续子数组的最大和 记得道题的技巧,就会做这道题,否则,就只能用暴力解了。 想到两种方法:1.暴力解,0(n^3);2.见牛客网解法,如果加到这个数字时的和还没有这个数字大,那么就把sum赋值为这个数字。 leetcode 152 乘积最大子数组 和连续子数组的最大和最大的区别是,不能用一个值来代表i之前的最大值,因为乘法有正负,所以需要一个max代表之前的最大值,min代表之前的最小值,然后包含当前元素的子数组的最大值为max(nums[i]*max,nums[i]*min,nums[i]),最小值为min(nums[i]*max,nums[i]*min,nums[i]),用res记录全局最大值,最后返回res即可。 leetcode139 单词拆分 典型的动态规划,dp[i]表示i位置之前的字符串可以被拆分,空字符串默认可以被拆分,所以dp[0]=true,然后判断dp[i]时,如果j(0<=j
leetcode198 打家劫舍 f(k) = max(f(k – 2) + A_k, f(k – 1)) f(k)第k户的最大收入。A_k第i户的收入,假设k=0和-1时f(k)为0,k从1开始 leetcode62. 不同路径 方法1是从arr[0,0]开始,不断向下或 向右移动,如果能到达右下角,加1,但是这个方法时间复杂度太高; 还有一种方法是动态规划,上边一行和左边一列都是0,且arr[i][j] = arr[i-1][j]+arr[i][j-1];;这样双重循环就可得到结果。 leetcode63 最小路径和 从右下角到左上角,grid(i,j)=grid(i,j)+min(grid(i+1,j),grid(i,j+1)) leetcode96 不同的二叉搜索树 典序的动态规划,G(n)=G(0)*G(n-1)+G(1)*G(n-2)+...+G(n-1)*G(0),双重循环,不断计算G(0),G(1),G(2)...G(n-1),直到G(n)。 leetcode 乘积最大数组 leetcode221 最大正方形 动态规划递推思想为,定义一个矩阵,dp,与原矩阵规模相同,dp[i][j]代表以这个位置为右下角的正方形的面积,这样的话,上边和左边值为1的位置dp值为1,其余值为1的位置dp值为左、上、左上位置的值中最小的加1。 leetcode279 完全平方数 动态规划递推思想为,f(x) = f(x-k)+1 k属于小于等于x的平方数的集合,递归的话时间复杂度较高,所以用动态规划,定义一个n+1长度的数组,dp[0]=0,fp[n]=dp[n-k]+1,这样就不用递归了。 类型12 全排列 leetcode78 子集 一开始res中只有[], [[]],然后遍历arr,每次在原有子集的基础上,新增一个[[], [1]],之后[[],[1],[2],[1,2]],然后[[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]]。 类型13 图 leetcode 207 课程表 学到了:用List 类型14 贪心算法 leetcode300 最长上升子序列 贪心算法+二分查找 最长上升子序列,最好能保持上升序列中数字最小,这是一种贪心思想,具体来做,就是遍历数组,如果数组中的数比先行序列中最大数字大,子序列长度加1,否则找到第一个比它小的数字(二分查找),然后将这个数字的后一个数字替换掉。 时间复杂度nlogn 也可以用动态规划,dp[i]记录以i元素结尾位置上升子序列的值,时间复杂度n^2。 leetcode 类型15 其他(5) 12. 数值的整数次方 1)用位运算计算数字的正整数次幂 2)base=0时,指数不能为负 3)base小于0,求 倒数 19. 顺时针打印矩阵 关键点1:层数 2*level 关键点2:上边(不管啥样都有) 右边(最少两列) 下边(最少两行两列) 左边(最少三号两列) 31. 整数中1出现的次数 偷奸取巧,把数字转化为字符串,然后遍历 32. 丑数 记方法吧,记住了就会,记不住就不会 45. 扑克牌顺子(抽象建模能力) 关键点1:除0以外不能有其他数字出现两次 关键点2:max-min<=4,不管有几个0 leetcode146.LRU缓存机制 第一点是LRU这种数据结构的特征挺有意思的,第二点是用HashMap与双向链表构造这一数据结构的过程挺有意思的 leetcode208 实现Trie(前缀树) 前缀树也称字典树,在敏感词过滤和词语补全等方面很有用,用result数组表示26个字母在当前节点是否出现,用isStirng表示当前节点是否某个单词的结尾单词。
是出现一次的。Integer.min()很好用。导入HashMap不能直接用Map,还得导入Map。反之也不可以
2)回到上一层时,将arrayList中最后一个元素移除PriorityQueue
判断某数字最后一位是否为1,num & 1 == 1,true为1,否则为0;是否为0 num & 1 == 0,true为0,否则为1。
数组中只出现一次的数字如果只有一个,其他数字出现偶数次,那么更好算了,异或一次就可以了。> arr来做邻接矩阵,还不是稀疏的,挺好。用indegres表示入度数组,用queue表示入度为0的点(无限制学习的课程)。这道题本质上是判断图是否有环。