剑指Offer刷题总结

Posted kyrie17

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了剑指Offer刷题总结相关的知识,希望对你有一定的参考价值。

1. 二维数组中的查找

在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。

思路题

从右上角或者左下角开始缩小范围。

2. 替换空格

请实现一个函数,将一个字符串中的每个空格替换成“%20”。例如,当字符串为We Are Happy.则经过替换之后的字符串为We%20Are%20Happy。

语法题

3. 从尾到头打印链表

输入一个链表,按链表从尾到头的顺序返回一个ArrayList。

头插法

数组反转

递归

4. 重建二叉树

输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树。假设输入的前序遍历和中序遍历的结果中都不含重复的数字。例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6},则重建二叉树并返回。

递归

API:Arrays.copyOfRange()

5. 用两个栈实现队列

用两个栈来实现一个队列,完成队列的Push和Pop操作。 队列中的元素为int类型。

简单题

6. 旋转数组的最小数字

把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。
输入一个非递减排序的数组的一个旋转,输出旋转数组的最小元素。
例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1。
NOTE:给出的所有元素都大于0,若数组大小为0,请返回0。

二分找到旋转点

7. 斐波那契数列

大家都知道斐波那契数列,现在要求输入一个整数n,请你输出斐波那契数列的第n项(从0开始,第0项为0)。
n<=39

优化成非递归

8. 跳台阶

一只青蛙一次可以跳上1级台阶,也可以跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法(先后次序不同算不同的结果)。

斐波那契数列

9. 变态跳台阶

一只青蛙一次可以跳上1级台阶,也可以跳上2级……它也可以跳上n级。求该青蛙跳上一个n级的台阶总共有多少种跳法。

递归

因为n级台阶,第一步有n种跳法:跳1级、跳2级、到跳n级
跳1级,剩下n-1级,则剩下跳法是f(n-1)
跳2级,剩下n-2级,则剩下跳法是f(n-2)
所以f(n)=f(n-1)+f(n-2)+...+f(1)
因为f(n-1)=f(n-2)+f(n-3)+...+f(1)所以f(n)=2*f(n-1)

10. 矩阵覆盖

我们可以用2*1的小矩形横着或者竖着去覆盖更大的矩形。请问用n个2*1的小矩形无重叠地覆盖一个2*n的大矩形,总共有多少种方法?

比如n=3时,2*3的矩形块有3种覆盖方法:
技术图片

递归

11. 二进制中1的个数

输入一个整数,输出该数二进制表示中1的个数。其中负数用补码表示。

语法题

12. 数值的整数次方

给定一个double类型的浮点数base和int类型的整数exponent。求base的exponent次方。
保证base和exponent不同时为0

语法题

13. 调整数组顺序使奇数位于偶数前面

输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变。

插入排序

时间换空间

新建一个数组

空间换时间

14. 链表中倒数第k个结点

输入一个链表,输出该链表中倒数第k个结点。

双指针

15. 反转链表

输入一个链表,反转链表后,输出新链表的表头。

语法题

16. 合并两个排序的链表

输入两个单调递增的链表,输出两个链表合成后的链表,当然我们需要合成后的链表满足单调不减规则。

语法题

17. 树的子结构

输入两棵二叉树A,B,判断B是不是A的子结构。(ps:我们约定空树不是任意一个树的子结构)

递归

18. 二叉树的镜像

操作给定的二叉树,将其变换为源二叉树的镜像。

           8
           /            6   10
         /   /         5  7 9 11
        镜像二叉树
            8
           /            10   6
         /   /         11 9 7  5

先根遍历

19. 顺时针打印矩阵

输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数字1,2,3,4,8,12,16,15,14,13,9,5,6,7,11,10.

通过四个指针定界

20. 包含min函数的栈

定义栈的数据结构,请在该类型中实现一个能够得到栈中所含最小元素的min函数(时间复杂度应为O(1))。
注意:保证测试中不会当栈为空的时候,对栈调用pop()或者min()或者top()方法。

两个栈

这里设计很巧妙:
push(),两个栈都要添加新元素;
pop()的时候两个栈都删元素。

21. 栈的压入、弹出序列

输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否可能为该栈的弹出顺序。假设压入栈的所有数字均不相等。例如序列1,2,3,4,5是某栈的压入顺序,序列4,5,3,2,1是该压栈序列对应的一个弹出序列,但4,3,5,1,2就不可能是该压栈序列的弹出序列。(注意:这两个序列的长度是相等的)

模拟栈

22. 从上往下打印二叉树

从上往下打印出二叉树的每个节点,同层节点从左至右打印。

队列

23. 二叉搜索树的后序遍历序列

输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历的结果。如果是则输出Yes,否则输出No。假设输入的数组的任意两个数字都互不相同。

递归

BST的后序序列的合法序列是,对于一个序列S,最后一个元素是x (也就是根),如果去掉最后一个元素的序列为T,那么T满足:T可以分成两段,前一段(左子树)小于x,后一段(右子树)大于x,且这两段(子树)都是合法的后序序列。完美的递归定义 : ) 。

24. 二叉树中和为某一值的路径

输入一颗二叉树的根节点和一个整数,打印出二叉树中结点值的和为输入整数的所有路径。路径定义为从树的根结点开始往下一直到叶结点所经过的结点形成一条路径。(注意: 在返回值的list中,数组长度大的数组靠前)

前序遍历

25. 复杂链表的复制

输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的head。(注意,输出结果中请不要返回参数中的节点引用,否则判题程序会直接返回空)

hashmap

  1. 遍历原链表,通过hashMap保存原链表节点映射复制链表节点
  2. 遍历新链表,找到对应的random

26. 二叉搜索树与双向链表

输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表。要求不能创建任何新的结点,只能调整树中结点指针的指向。

中序遍历

27. 字符串的排序

数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}。由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2。如果不存在则输出0。

递归

for (int i = begin; i < arr.length; i++) {
    swap(arr, begin, i);
    permutation(arr, begin + 1);
    swap(arr, begin, i);
}

28. 数组中出现次数超过一半的数字

数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}。由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2。如果不存在则输出0。

hashmap

29. 最小的k个数

输入n个整数,找出其中最小的K个数。例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,。

最大堆

30. 连续子数组的最大和

HZ偶尔会拿些专业问题来忽悠那些非计算机专业的同学。今天测试组开完会后,他又发话了:在古老的一维模式识别中,常常需要计算连续子向量的最大和,当向量全为正数的时候,问题很好解决。但是,如果向量中包含负数,是否应该包含某个负数,并期望旁边的正数会弥补它呢?例如:{6,-3,-2,7,-15,1,2,2},连续子向量的最大和为8(从第0个开始,到第3个为止)。给一个数组,返回它的最大连续子序列的和,你会不会被他忽悠住?(子向量的长度至少是1)

动态规划

31. 整数中1出现的次数

求出1~13的整数中1出现的次数,并算出100~1300的整数中1出现的次数?为此他特别数了一下1~13中包含1的数字有1、10、11、12、13因此共出现6次,但是对于后面问题他就没辙了。ACMer希望你们帮帮他,并把问题更加普遍化,可以很快的求出任意非负整数区间中1出现的次数(从1 到 n 中1出现的次数)。

按位分析

主要思想:对每个数位上有多少包含1的点进行分析。

根据设定的整数位置,对n进行分割,分为两部分,高位n/i,低位n%i

下面我们通过考虑 n 的百位对应的数的情况,来分析:

  1. 当i表示百位,且百位对应的数 >= 2,如n = 31456,i = 100,则a = 314,b = 56,此时百位为1的次数有a/10+1=32(最高两位0~31),每一次都包含100个连续的点,即共有(a/10+1)*100个点的百位为1。
    技术图片

  2. 当i表示百位,且百位对应的数为1,如n = 31156,i = 100,则a = 311,b = 56,此时百位对应的就是1,则共有a/10=31(最高两位0-30)次是包含100个连续点,当最高两位为31(即a=311),本次只对应局部点00~56,共b+1次,所有点加起来共有(a/10*100)+(b+1),这些点百位对应为1。
  3. 当i表示百位,且百位对应的数为0,如n = 31056,i = 100,则a = 310,b = 56,此时百位为1的次数有a/10=31(最高两位0~30)。

综合以上三种情况,当百位对应0或>=2时,有(a+8)/10次包含所有100个点,还有当百位为1(a%10==1),需要增加局部点b+1。

之所以补8,是因为当百位为0,则a/10==(a+8)/10,当百位>=2,补8会产生进位位,效果等同于(a/10+1)。

// 统计次数
int count = 0;
for(int i = 1; i <= n; i *= 10){
    // 计算高位和低位
    int a = n / i, b = n % i;
    count += (a + 8) / 10 * i + (a % 10 == 1 ? 1 : 0) * (b + 1);
}
return count;

有人可能会有疑惑,比如11100,这个数在考虑百位为1的时候算作了一次,在考虑千位的时候也算了一次,在考虑万位为1的时候又算了一次,一共计了3次,这不是明显重复吗?

其实并不重复,题目中要我们统计出现的1的个数,那么我们可以看到11100一共是3个1,如果剔除了重复的情况只考虑一次才会是问题。换言之,在计算从1到n整数中1的出现次数时,我们把10位出现1的情况个数加上百位出现1的情况个数一直加到最高位是1的情况的个数,这里面一个数可能被统计过多次;11100百位出现1,千位和万位都为1,那么被重复统计了3次。

32. 把数组排成最小的数

输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个。例如输入数组{3,32,321},则打印出这三个数字能排成的最小数字为321323。

字典序

改写compare()

Arrays.sort(str,new Comparator<String>(){
    @Override
    public int compare(String s1, String s2) {
        String c1 = s1 + s2;
        String c2 = s2 + s1;
        return c1.compareTo(c2);
    }
});

33. 丑数

把只包含质因子2、3和5的数称作丑数(Ugly Number)。例如6、8都是丑数,但14不是,因为它包含质因子7。 习惯上我们把1当做是第一个丑数。求按从小到大的顺序的第N个丑数。

思路题

一个丑数的因子只有2,3,5,那么丑数p = 2 ^ x * 3 ^ y * 5 ^ z

那么我们从1开始乘以2,3,5,就得到2,3,5三个丑数,在从这三个丑数出发乘以2,3,5就得到4,6,10,6,9,15,10,15,25九个丑数,这里我们发现会得到重复的丑数,而且得到的丑数也是无序的。

那么我们可以维护三个队列:

  • 乘以2的队列
  • 乘以3的队列
  • 乘以5的队列
// 选出三个队列头最小的数
res[i] = Math.min(res[p2] * 2, Math.min(res[p3] * 3, res[p5] * 5));
// 这三个if有可能进入一个或者多个,进入多个是三个队列头最小的数有多个的情况
if (res[i] == res[p2] * 2) p2++;
if (res[i] == res[p3] * 3) p3++;
if (res[i] == res[p5] * 5) p5++;

34. 第一个只出现一次的字符

在一个字符串(0<=字符串长度<=10000,全部由字母组成)中找到第一个只出现一次的字符,并返回它的位置, 如果没有则返回 -1(需要区分大小写)。

hashmap

35. 数组中的逆序对

在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数P。并将P对1000000007取模的结果输出。 即输出P%1000000007

归并排序

36. 两个链表的第一个公共结点

输入两个链表,找出它们的第一个公共结点。

堆栈

  1. 把两个链表放入两个栈,弹出,直到结果不相等为止
  2. 最后一个相等的就是第一个公共结点

双指针

假设list1长度为a + n,list2 长度为b + n,a < b

  1. p1会先走到list1尾部,将p1置为list2头部
  2. p2接着走到list2尾部,将p2置为list1头部
  3. 在p1和p2都走了a+b+n步后,p1,p2到达 null 或者公共结点

37. 数字在排序数组中出现的次数

统计一个数字在排序数组中出现的次数。

二分

二分找到数组中第一个出现该数字下标和最后一个出现该数字的下标

38. 二叉树的深度

输入一棵二叉树,求该树的深度。从根结点到叶结点依次经过的结点(含根、叶结点)形成树的一条路径,最长路径的长度为树的深度。

递归

递归比较左右子树

39. 平衡二叉树

输入一棵二叉树,判断该二叉树是否是平衡二叉树。

递归

递归左右子树差的绝对值小于等于1

40. 数组中只出现一次的数字

一个整型数组里除了两个数字之外,其他的数字都出现了两次。请写程序找出这两个只出现一次的数字。

位运算

只有一个数出现一次时,我们把数组中所有的数,依次异或运算,最后剩下的就是落单的数,因为成对出现的都抵消了。

对于两个数(我们假设是A、B)出现一次的数组:

  1. 还是先异或,剩下的数字为A、B异或的结果(这个结果的二进制中的1,表现的是A和B的不同的位)
  2. 从右取第一个1所在的位数
  3. 这样可以将原数组分成两组,A、B肯定不在一组
  4. 问题就回到了“数组只有一个数出现一次”的问题了,再次异或,剩余的两个结果就是这两个只出现一次的数字。

41. 和为S的连续正数序列

小明很喜欢数学,有一天他在做数学作业时,要求计算出9~16的和,他马上就写出了正确答案是100。但是他并不满足于此,他在想究竟有多少种连续的正数序列的和为100(至少包括两个数)。没多久,他就得到另一组连续正数和为100的序列:18,19,20,21,22。现在把问题交给你,你能不能也很快的找出所有和为S的连续正数序列? Good Luck!

双指针

  1. 两个起点,相当于动态窗口的两边,根据其窗口内的值的和来确定窗口的位置和大小
  2. 当前窗口内的值之和是(a0+an)*n/2
  3. 如果和sum相等,那么就将窗口范围的所有数添加进结果集
  4. 如果小于sum,那么右边窗口右移一下
  5. 如果当前窗口内的值之和大于sum,那么左边窗口右移一下

42. 和为S的两个数字

输入一个递增排序的数组和一个数字S,在数组中查找两个数,使得他们的和正好是S,如果有多对数字的和等于S,输出两个数的乘积最小的。

双指针(相差越远乘积越小)

  1. 数列满足递增,设两个头尾两个指针i和j
  2. 若ai + aj == sum,就是答案(相差越远乘积越小)
  3. 若ai + aj > sum,aj肯定不是答案之一(前面已得出 i 前面的数已是不可能),j--
  4. 若ai + aj < sum,ai肯定不是答案之一(前面已得出 j 后面的数已是不可能),i++

43. 左旋转字符串

汇编语言中有一种移位指令叫做循环左移(ROL),现在有个简单的任务,就是用字符串模拟这个指令的运算结果。对于一个给定的字符序列S,请你把其循环左移K位后的序列输出。例如,字符序列S=”abcXYZdef”,要求输出循环左移3位后的结果,即“XYZdefabc”。是不是很简单?OK,搞定它!

substring

两次翻转字符串

主要思想是:

YX = (X^{T}Y^{T})^{T}
  1. 先反转前面部分字符串,再反转后面部分字符串
  2. 反转整个字符串

44. 翻转单词顺序列

牛客最近来了一个新员工Fish,每天早晨总是会拿着一本英文杂志,写些句子在本子上。同事Cat对Fish写的内容颇感兴趣,有一天他向Fish借来翻看,但却读不懂它的意思。例如,“student. a am I”。后来才意识到,这家伙原来把句子单词的顺序翻转了,正确的句子应该是“I am a student.”。Cat对一一的翻转这些单词顺序可不在行,你能帮助他么?

堆栈

先翻转整个句子,再依次翻转每个单词

public class Solution {
????public String ReverseSentence(String str) {
????????char[] chars = str.toCharArray();
????????reverse(chars, 0, chars.length - 1);
????????int blank = -1;
????????for(int i = 0; i < chars.length; i++){
????????????if(chars[i] == ' '){?
????????????????int nextBlank = i;
????????????????reverse(chars, blank + 1, nextBlank - 1);
????????????????blank = nextBlank;
????????????}
????????}
????????reverse(chars, blank + 1, chars.length - 1);//最后一个单词单独进行反转
????????return new String(chars);
?????????
????}
????public void reverse(char[] chars, int low, int high){
????????while(low < high){
????????????char temp = chars[low];
????????????chars[low] = chars[high];
????????????chars[high] = temp;
????????????low++;
????????????high--;
????????}
????}
}

45. 扑克牌顺子

LL今天心情特别好,因为他去买了一副扑克牌,发现里面居然有2个大王,2个小王(一副牌原本是54张^_^)...他随机从中抽出了5张牌,想测测自己的手气,看看能不能抽到顺子,如果抽到的话,他决定去买体育彩票,嘿嘿!!“红心A,黑桃3,小王,大王,方片5”,“Oh My God!”不是顺子.....LL不高兴了,他想了想,决定大小 王可以看成任何数字,并且A看作1,J为11,Q为12,K为13。上面的5张牌就可以变成“1,2,3,4,5”(大小王分别看作2和4),“So Lucky!”。LL决定去买体育彩票啦。 现在,要求你使用这幅牌模拟上面的过程,然后告诉我们LL的运气如何, 如果牌能组成顺子就输出true,否则就输出false。为了方便起见,你可以认为大小王是0。

思路题

  1. max - min < 5
  2. 除0外没有重复的数字

46. 孩子们的游戏(圆圈中最后剩下的数)

每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此。HF作为牛客的资深元老,自然也准备了一些小游戏。其中,有个游戏是这样的:首先,让小朋友们围成一个大圈。然后,他随机指定一个数m,让编号为0的小朋友开始报数。每次喊到m-1的那个小朋友要出列唱首歌,然后可以在礼品箱中任意的挑选礼物,并且不再回到圈中,从他的下一个小朋友开始,继续0...m-1报数....这样下去....直到剩下最后一个小朋友,可以不用表演,并且拿到牛客名贵的“名侦探柯南”典藏版(名额有限哦!!^_^)。请你试着想下,哪个小朋友会得到这份礼品呢?(注:小朋友的编号是从0到n-1)

如果没有小朋友,请返回-1

利用数组来模拟环

  1. 用一个数组做标志
  2. if(i>=n) i=0; 模拟环?

47. 求1+2+3+...+n

求1+2+3+...+n,要求不能使用乘除法、for、while、if、else、switch、case等关键字及条件判断语句(A?B:C)。

递归

可以利用&&判断递归出口

boolean flag = (n > 0) && ((sum += Sum_Solution(n - 1)) > 0);

48. 不用加减乘除做加法

写一个函数,求两个整数之和,要求在函数体内不得使用+、-、*、/四则运算符号。

与或运算

  1. 相加各位的值,不算进位
  2. 计算进位值
  3. 重复上述两步,直至进位值为0

49. 把字符串转换成整数

将一个字符串转换成一个整数,要求不能使用字符串转换整数的库函数。 数值为0或者字符串不是一个合法的数值则返回0

溢出判断

https://blog.csdn.net/qq_33330687/article/details/81626157

50. 数组中重复的数字

在一个长度为n的数组里的所有数字都在0到n-1的范围内。 数组中某些数字是重复的,但不知道有几个数字是重复的。也不知道每个数字重复几次。请找出数组中任意一个重复的数字。 例如,如果输入长度为7的数组{2,3,1,0,2,5,3},那么对应的输出是第一个重复的数字2。

利用现有数组

使用hashmap有空间消耗,利用现有数组设置标志,当一个数字被访问过后,可以设置对应位上的数 + n,之后再遇到相同的数时,会发现对应位上的数已经大于等于n了,那么直接返回这个数即可。

51. 构建乘积数组

给定一个数组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]。不能使用除法。

矩阵连乘

技术图片

下三角用连乘可以很容求得,上三角,从下向上也是连乘。

52. 正则表达式匹配

请实现一个函数用来匹配包括‘.‘和‘*‘的正则表达式。模式中的字符‘.‘表示任意一个字符,而‘*‘表示它前面的字符可以出现任意次(包含0次)。 在本题中,匹配是指字符串的所有字符匹配整个模式。例如,字符串"aaa"与模式"a.a"和"ab*ac*a"匹配,但是与"aa.a"和"ab*a"均不匹配

思路题

分为以下的情况:

  1. 模式第二个为*
    • 字符串第一个和模式第一个匹配
      • 匹配0个字符
      • 匹配1个字符
      • 先匹配str中1个字符,再匹配str中的下一个
    • 字符串第一个和模式第一个不匹配
  2. 模式第二个不为*
    • 字符串第1个跟模式第1个匹配
    • 字符串第1个跟模式第1个不匹配

53. 表示数值的字符串

请实现一个函数用来判断字符串是否表示数值(包括整数和小数)。例如,字符串"+100","5e2","-123","3.1416"和"-1E-16"都表示数值。 但是"12e","1a3.14","1.2.3","+-5"和"12e+4.3"都不是。

思路题

通过三个变量index(此处是否允许有+/-),small(此处是否允许有.),next(此处是否需要后面不为空)控制。

54. 字符流中第一个不重复的字符

请实现一个函数用来找出字符流中第一个只出现一次的字符。例如,当从字符流中只读出前两个字符"go"时,第一个只出现一次的字符是"g"。当从该字符流中读出前六个字符“google"时,第一个只出现一次的字符是"l"。

简单题

一个字符占8位,因此不会超过256个,可以申请一个256大小的数组来实现一个简易的哈希表。

55. 链表中环的入口结点

给一个链表,若其中包含环,请找出该链表的环的入口结点,否则,输出null。

快慢指针

主要的思想是:用两个指针 first,second 分别从起点开始走,first 每次走一步,second 每次走两步。
如果过程中 second 走到null,则说明不存在环。否则当 first 和 second 相遇后,让 first 返回起点,second 待在原地不动,然后两个指针每次分别走一步,当相遇时,相遇点就是环的入口。

技术图片

如上图所示,a 是起点,b 是环的入口,c 是两个指针的第一次相遇点,ab 之间的距离是 x,bc 之间的距离是 y。
则当 first 走到 b 时,由于 second 比 first 多走一倍的路,所以 second 已经从 b 开始在环上走了 x 步,可能多余1圈,距离 b 还差 y 步(这是因为第一次相遇点在 b 之后 y 步,我们让 first 退回 b 点,则 second 会退 2y 步,也就是距离 b 点还差 y 步);所以 second 从 b 点走 x+y 步即可回到 b 点,所以 second 从 c 点开始走,走 x 步即可恰好走到 b 点,同时让 first 从头开始走,走 x 步也恰好可以走到 b 点。所以第二次相遇点就是 b 点。

56. 删除链表中重复的结点

在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针。 例如,链表1->2->3->3->4->4->5 处理后为 1->2->5

双指针

利用零结点,减少首结点的处理。

57. 二叉树的下一个结点

给定一个二叉树和其中的一个结点,请找出中序遍历顺序的下一个结点并且返回。注意,树中的结点不仅包含左右子结点,同时包含指向父结点的指针。

简单题

  1. 判断是否有右子节点
  2. 没有的话找到该结点的父节点

58. 对称的二叉树

请实现一个函数,用来判断一颗二叉树是不是对称的。注意,如果一个二叉树同此二叉树的镜像是同样的,定义其为对称的。

简单题

技术图片

左右等于右左,左左等于右右

59. 按之字形顺序打印二叉树

请实现一个函数按照之字形打印二叉树,即第一行按照从左到右的顺序打印,第二层按照从右至左的顺序打印,第三行按照从左到右的顺序打印,其他行以此类推。

两个栈

  1. s1存奇数层结点
  2. s2存偶数层结点

60. 把二叉树打印成多行

从上到下按层打印二叉树,同一层结点从左至右输出。每一层输出一行。

队列

61. 序列化二叉树

请实现两个函数,分别用来序列化和反序列化二叉树

二叉树的序列化是指:把一棵二叉树按照某种遍历方式的结果以某种格式保存为字符串,从而使得内存中建立起来的二叉树可以持久保存。序列化可以基于先序、中序、后序、层序的二叉树遍历方式来进行修改,序列化的结果是一个字符串,序列化时通过 某种符号表示空节点(#),以 ! 表示一个结点值的结束(value!)。
二叉树的反序列化是指:根据某种遍历顺序得到的序列化字符串结果str,重构二叉树。

递归

通过先根遍历将二叉树转换成字符串,通过全局变量index递归完成而二叉树的重建。

62. 二叉搜索树的第k个结点

给定一棵二叉搜索树,请找出其中的第k小的结点。例如, (5,3,7,2,4,6,8)? ? 中,按结点数值大小顺序第三小结点的值为4。

中序遍历

63. 数据流中的中位数

如何得到一个数据流中的中位数?如果从数据流中读出奇数个数值,那么中位数就是所有数值排序之后位于中间的数值。如果从数据流中读出偶数个数值,那么中位数就是所有数值排序之后中间两个数的平均值。我们使用Insert()方法读取数据流,使用GetMedian()方法获取当前读取数据的中位数。

大、小顶堆

主要的思想是:因为要求的是中位数,那么这两个堆,大顶堆用来存较小的数,从大到小排列;小顶堆存较大的数,从小到大的顺序排序,显然中位数就是大顶堆的根节点与小顶堆的根节点和的平均数。

  1. 通过PriorityQueue创建大顶堆和小顶堆:
//小顶堆
private PriorityQueue<Integer> minHeap = new PriorityQueue<>();
//大顶堆
private PriorityQueue<Integer> maxHeap = new PriorityQueue<Integer>(15, new Comparator<Integer>() {
    @Override
    public int compare(Integer o1, Integer o2) {
        return o2 - o1;
    }
});
  1. 当数目为偶数的时候,将这个值插入大顶堆中,再将大顶堆中根节点(即最大值)插入到小顶堆中
  2. 当数目为奇数的时候,将这个值插入小顶堆中,再讲小顶堆中根节点(即最小值)插入到大顶堆中
  3. 取中位数的时候,如果当前个数为偶数,显然是取小顶堆和大顶堆根结点的平均值;如果当前个数为奇数,显然是取小顶堆的根节点

64. 滑动窗口的最大值

给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值。例如,如果输入数组{2,3,4,2,6,2,5,1}及滑动窗口的大小3,那么一共存在6个滑动窗口,他们的最大值分别为{4,4,6,6,6,5}; 针对数组{2,3,4,2,6,2,5,1}的滑动窗口有以下6个: {[2,3,4],2,6,2,5,1}, {2,[3,4,2],6,2,5,1}, {2,3,[4,2,6],2,5,1}, {2,3,4,[2,6,2],5,1}, {2,3,4,2,[6,2,5],1}, {2,3,4,2,6,[2,5,1]}。

双端队列

使用双端队列,记录每个窗口的最大值下标

  1. 先删去队列中num[index]比num[i]小的
  2. 将i压入队列尾部
  3. 判断队首元素是否过期(即是否在给定窗口中)
LinkedList<Integer> qmax = new LinkedList<>();

65. 矩阵中的路径

请设计一个函数,用来判断在一个矩阵中是否存在一条包含某字符串所有字符的路径。路径可以从矩阵中的任意一个格子开始,每一步可以在矩阵中向左,向右,向上,向下移动一个格子。如果一条路径经过了矩阵中的某一个格子,则该路径不能再进入该格子。 例如 a b c e s f c s a d e e 矩阵中包含一条字符串"bcced"的路径,但是矩阵中不包含"abcb"路径,因为字符串的第一个字符b占据了矩阵中的第一行第二个格子之后,路径不能再次进入该格子。

回溯法

  1. 找到矩阵中起点等于给定字符串第一个元素的值
  2. 回溯

66. 机器人的运动范围

地上有一个m行和n列的方格。一个机器人从坐标0,0的格子开始移动,每一次只能向左,右,上,下四个方向移动一格,但是不能进入行坐标和列坐标的数位之和大于k的格子。 例如,当k为18时,机器人能够进入方格(35,37),因为3+5+3+7 = 18。但是,它不能进入方格(35,38),因为3+5+3+8 = 19。请问该机器人能够达到多少个格子?

回溯法

67. 剪绳子

给你一根长度为n的绳子,请把绳子剪成整数长的m段(m、n都是整数,n>1并且m>1),每段绳子的长度记为k[0],k[1],...,k[m]。请问k[0]xk[1]x...xk[m]可能的最大乘积是多少?例如,当绳子的长度是8时,我们把它剪成长度分别为2、3、3的三段,此时得到的最大乘积是18。

输入:target(为绳子长度)
输出:最大乘积

动态规划

在这个题目中,由于必须要剪一刀,因此会导致当所给的绳子长度是小于4的时候,剪完之后的乘积小于剪之前的长度。

if (target < 2) return 0;
if (target == 2) return 1;
if (target == 3) return 2;

大于等于4可以分成很多段,比如4可分成1和3,这里的3不再分,因为3分段最大才2,不继续分就是3,当然4也可以分成2和2,乘积为4,记录最大的。

int[] dp = new int[target + 1];
dp[1] = 1;
dp[2] = 2;
dp[3] = 3;
int res = 0;
for (int i = 4; i < target + 1; i++) {
for (int j = 1; j <= i / 2; j++) {
    res = Math.max(res, dp[j] * dp[i - j]);
}
dp[i] = res;
}
return dp[target];

贪心算法

当n>=5的时候,2(n-2) > n and 3(n-3) > n and 3(n-3) >= 2(n-2)
也就是说当长度大于5的时候,剪出长度为3的段数越多最后的乘积越大

但是对于长度为4的时候,如果剪出一段长度为3,另一段长度为1,则乘积为3,小于剪出两段长度为2,因此,当最后剩下4的时候,不是剪成1,3而是剪成2,2

if (target < 2) return 0;
if (target == 2) return 1;
if (target == 3) return 2;
int?x?=?target?%?3;
int?y?=?target?/?3;
if?(x?==?0)?{
    return?Math.pow(3,?y);
}?else?if?(x?==?1)?{    // 最后剩下4的时候,不是剪成1,3而是剪成2,2
    return?2?*?2?* Math.?pow(3,?y?-?1);
}?else?{    // 最后剩下2
    return?2?*??Math.pow(3,?y);
}

以上是关于剑指Offer刷题总结的主要内容,如果未能解决你的问题,请参考以下文章

剑指 offer 刷题记录

java刷题--剑指offer05 替换空格

java刷题-剑指offer 07 重建二叉树

剑指offer系列刷题第一篇——寻找单身狗

java刷题-剑指offer 12 矩阵中的路径

java刷题-剑指offer 11 旋转数组的最小数字