链表是咱们数据结构面试中比较容易出错的问题,因此不少面试官总喜欢在这上面下功夫,为了不出错,咱们最好先进行全面的分析。在实际软件开发周期中,设计的时间一般不会比编码的时间短,在面试的时候咱们不要着急于写代码,而是一开始仔细分析和设计,这将给面试官留下一个很好的印象。java
与其很快写出一段千疮百孔的代码,不容仔细分析后再写出健壮性无敌的程序。node
面试题:输入一个单链表的头结点,返回它的中间元素。为了方便,元素值用整型表示。面试
当应聘者看到这道题的时候,心里一阵狂喜,怎么给本身遇到了这么简单的题。拿起笔就开始写,先遍历整个链表,拿到链表的长度 len,再次遍历链表,位于 len/2 的元素就是链表的中间元素。算法
因此这个题最重要的点就是拿到链表的长度 len。而拿到这个 len 也比较简单,只须要遍历前设定一个 count 值,遍历的时候 count++ ,第一次遍历结束,就拿到单链表的长度 len 了。微信
因而咱们很快写出了这样的代码:数据结构
public class Test15 {
public static class LinkNode {
int data;
LinkNode next;
public LinkNode(int data) {
this.data = data;
}
}
private static int getTheMid(LinkNode head) {
int count = 0;
LinkNode node = head;
while (head != null) {
head = head.next;
count++;
}
for (int i = 0; i < count / 2; i++) {
node = node.next;
}
return node.data;
}
public static void main(String[] args) {
LinkNode head = new LinkNode(1);
head.next = new LinkNode(2);
head.next.next = new LinkNode(3);
head.next.next.next = new LinkNode(4);
head.next.next.next.next = new LinkNode(5);
System.out.println(getTheMid(head));
}
}
复制代码
面试官看到这个代码的时候,他告诉咱们上面代码循环了两次,可是他期待的只有一次。学习
因而咱们绞尽脑汁,忽然想到了网上介绍过的一个概念:快慢指针法。this
假设咱们设置两个变量 slow、fast 起始都指向单链表的头结点当中,而后依次向后面移动,fast 的移动速度是 slow 的 2 倍。这样当 fast 指向末尾节点的时候,slow 就正好在正中间了。编码
想清楚这个思路后,咱们很快就能写出以下代码:spa
public class Test15 {
public static class LinkNode {
int data;
LinkNode next;
public LinkNode(int data) {
this.data = data;
}
}
private static int getTheMid(LinkNode head) {
LinkNode slow = head;
LinkNode fast = head;
while (fast != null && fast.next != null) {
fast = fast.next.next;
slow = slow.next;
}
return slow.data;
}
public static void main(String[] args) {
LinkNode head = new LinkNode(1);
head.next = new LinkNode(2);
head.next.next = new LinkNode(3);
head.next.next.next = new LinkNode(4);
head.next.next.next.next = new LinkNode(5);
System.out.println(getTheMid(head));
}
}
复制代码
快慢指针法 确实在链表类面试题中特别好用,咱们不妨在这里触类旁通,对原题稍微修改一下,其实也能够实现。
面试题:给定一个单链表的头结点,判断这个链表是不是循环链表。
和前面的问题同样,咱们只须要定义两个变量 slow,fast,同时从链表的头结点出发,fast 每次走链表,而 slow 每次只走一步。若是走得快的指针追上了走得慢的指针,那么链表就是环形(循环)链表。若是走得快的指针走到了链表的末尾(fast.next 指向 null)都没有追上走得慢的指针,那么链表就不是环形链表。
有了这样的思路,实现代码那还不是分分钟的事儿。
public class Test15 {
public static class LinkNode {
int data;
LinkNode next;
public LinkNode(int data) {
this.data = data;
}
}
private static boolean isRingLink(LinkNode head) {
LinkNode slow = head;
LinkNode fast = head;
while (slow != null && fast != null && fast.next != null) {
if (slow == fast || fast.next = slow) {
return true;
}
fast = fast.next.next;
slow = slow.next;
}
return false;
}
public static void main(String[] args) {
LinkNode head = new LinkNode(1);
head.next = new LinkNode(2);
head.next.next = new LinkNode(3);
head.next.next.next = new LinkNode(4);
head.next.next.next.next = new LinkNode(5);
System.out.println(isRingLink(head));
head.next.next.next.next.next = head;
System.out.println(isRingLink(head));
}
}
复制代码
确实有意思,快慢指针法 再一次利用它的优点巧妙解决了咱们的问题。
咱们上面讲解的「快慢指针法」均是一个变量走 1 步,一个变量走 n 步。咱们其实还能够拓展它。这个「快慢」并非说必定要同时遍历。
好比《剑指Offer》中的第 15 道面试题,就运用到了「快慢指针法」的延展。
面试题:输入一个单链表的头结点,输出该链表中倒数第 k 个节点的值。
初一看这个彷佛并不像咱们前面学习到的「快慢指针法」的考察。因此大多数人就迷糊了,进入到常规化思考。依然仍是设置一个整型变量 count,而后每次循环的时候 count++,拿到链表的长度 n。那么倒数第 k 个节点也就是顺数第 n-k+1 个结点。因此咱们只须要在拿到长度 n 后再进行一次 n-k+1 次循环就能够拿到这个倒数第 k 个节点的值了。
但面试官显然不会太满意这个臃肿的解法,他依然但愿咱们一次循环就能搞定这个事。
为了实现只遍历一次链表就能找到倒数第 k 个结点,咱们依然能够定义两个遍历 slow 和 fast。咱们让 fast 变量先往前遍历 k-1 步,slow 保持不动。从第 k 步开始,slow 变量也跟着 fast 变量从链表的头结点开始遍历。因为两个变量指向的结点距离始终保持在 k-1,那么当 fast 变量到达链表的尾结点的时候,slow 变量指向的结点正好是咱们所须要的倒数第 k 个结点。
咱们依然能够在心中默认一遍代码:
在心中默走了一遍代码后,咱们显然很容易写出下面的代码。
public class Test15 {
public static class LinkNode {
int data;
LinkNode next;
public LinkNode(int data) {
this.data = data;
}
}
private static int getSpecifiedNodeReverse(LinkNode head, int k) {
LinkNode slow = head;
LinkNode fast = head;
if (fast == null) {
throw new RuntimeException("your linkNode is null");
}
// 先让 fast 先走 k-1 步
for (int i = 0; i < k - 1; i++) {
if (fast.next == null) {
// 说明输入的 k 已经超过了链表长度,直接报错
throw new RuntimeException("the value k is too large.");
}
fast = fast.next;
}
while (fast.next != null) {
slow = slow.next;
fast = fast.next;
}
return slow.data;
}
public static void main(String[] args) {
LinkNode head = new LinkNode(1);
head.next = new LinkNode(2);
head.next.next = new LinkNode(3);
head.next.next.next = new LinkNode(4);
head.next.next.next.next = new LinkNode(5);
System.out.println(getSpecifiedNodeReverse(head, 3));
System.out.println(getSpecifiedNodeReverse(null, 1));
}
}
复制代码
链表类面试题,真是能够玩出五花八门,当咱们用一个变量遍历链表不能解决问题的时候,咱们能够尝试用两个变量来遍历链表,可让其中一个变量遍历的速度快一些,好比一次走两步,或者是走若干步。咱们在遇到这类面试的时候,千万不要自乱阵脚,学会理性分析问题。
本来是想给个人小伙伴说再见了,但惟恐你们还没学到真本事,因此在这里再留一个拓展题。
面试题:给定一个单链表的头结点,删除倒数第 k 个结点。
哈哈,和上面的题目仅仅只是把得到它的值变成了删除,很多小伙伴确定都偷着乐了,但南尘仍是先提醒你们,不要太忘乎所以哟~
好啦,我们明天再见啦~
我是南尘,只作比心的公众号,欢迎关注我。