1

这周盘点一下面试中最容易被问到的链表类算法题,可能下一次面试中就会出现这些题目和技巧哦。

基础概念

首先,链表是一种常见的数据结构,常见的有单链表、双向链表等等。

拿单链表来举例,对于每一个节点可以使用下面的数据结构表示:

struct ListNode {
  val: any; // 节点的值
  next: ListNode; // 该节点指向的下一个节点
}

下图可以简单的描述一个链表的结构

对于链表来说,一定要掌握的操作就是添加节点和删除节点,因为这是所有技巧的基础。

删除节点

如果要在下图中删除2这个节点,就可以进行如下操作:

pre.next = cur.next;
cur.next = null;


因为需要遍历链表找到pre和cur,所以删除操作的时间复杂度是O(N),空间复杂度是O(1)

添加节点

如果要在下图中添加2这个节点,就可以进行如下操作

next = pre.next;
pre.next = cur;
cur.next = next;


添加新节点的时间复杂度是O(1),空间复杂度是O(1)

经典题目

反转链表

LeetCode.206,难度简单

题目

反转一个单链表。

示例:

输入: 1->2->3->4->5->NULL

输出: 5->4->3->2->1->NULL

思路

这道题目也算是链表类题目的老江湖了,印象中从校招一直考到社招,出现频率之高令人咋舌。

对于例子1->2->3->4->5->NULL来说,遍历一遍链表,把每个节点的next属性指向它的前一个节点即可,如下图所示:

对于每一个节点来说,需要知道它的前一个节点pre是谁,也需要知道它的下一个节点是谁(维持链表的遍历);下面我给出一个非递归的方法,当然也递归的方法,读者感兴趣可以自行实现一下。

代码
/**
 * Definition for singly-linked list.
 * function ListNode(val) {
 *     this.val = val;
 *     this.next = null;
 * }
 */
/**
 * @param {ListNode} head
 * @return {ListNode}
 */
var reverseList = function(head) {
    if(head === null || head.next === null)
      return head;
  
    let pre = null, cur = head;
    while(cur !== null) {
        const next = cur.next;
        cur.next = pre;
        pre = cur;
        cur = next;
    }
    
    return pre;
};

环形链表 II

题目

给定一个链表,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。

为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。 如果 pos 是 -1,则在该链表中没有环。

说明:不允许修改给定的链表。

示例 1:

输入:head = [3,2,0,-4], pos = 1

输出:tail connects to node index 1

解释:链表中有一个环,其尾部连接到第二个节点。

示例 2:

输入:head = [1,2], pos = 0

输出:tail connects to node index 0

解释:链表中有一个环,其尾部连接到第一个节点。

示例 3:

输入:head = [1], pos = -1

输出:no cycle

解释:链表中没有环。

思路

这也是一道经典的题目,可以使用快慢指针的办法来解决之,代码如下所示;

那么为什么使用快慢指针就可以检测出链表是否有环并且找到第一个入环节点呢?证明如下:

如图,设整个链表长度为L,环长度为R,且距离具有方向性,例如CB是C点到B点的距离,BC是B点到C点的距离,CB!=BC。当证明有环时,fast和slow都顺时针到了B点,则此时:

slow走的距离:AC+CB

fast走的距离:AC+k*R+CB(k=0,1,2...)

由于fast每次走2个节点,slow每次走1个节点,所以:

2(AC+CB) = AC+k*R+CB

AC+CB = k*R

AC+CB = (k-1)*R+R

AC = (k-1)*R+R-CB

AC = (k-1)*R+BC

从最终的表达式可以看出来,AC的距离等于绕环若干圈后再加上BC的距离,也就是说慢指针从A点出发以速度1前进、快指针从B点出发以速度1前进,则慢指针到C点时,快指针也必然到了。

代码
/**
 * Definition for singly-linked list.
 * function ListNode(val) {
 *     this.val = val;
 *     this.next = null;
 * }
 */

/**
 * @param {ListNode} head
 * @return {ListNode}
 */
var detectCycle = function(pHead) {
    if(pHead === null || pHead.next === null || pHead.next.next === null)
        return null;
    var fast = pHead;
    var slow = pHead;

    while(fast.next !==null && fast.next.next !== null) {
        slow = slow.next;
        fast = fast.next.next;
        if(slow === fast)
            break;
    }

    if(fast === null || fast.next === null || fast.next.next === null)
        return null;
    // 有环,slow重新回到链表头
    slow = pHead;
    
    // slow和fast重新相遇时,相遇节点就是入环节点
    while(slow !== fast) {
        slow = slow.next;
        fast = fast.next;
    }

    return slow;
};

奇偶链表

LeetCode.328,难度中等

题目

给定一个单链表,把所有的奇数节点和偶数节点分别排在一起。请注意,这里的奇数节点和偶数节点指的是节点编号的奇偶性,而不是节点的值的奇偶性。

请尝试使用原地算法完成。你的算法的空间复杂度应为 O(1),时间复杂度应为 O(nodes),nodes 为节点总数。

示例 1:

输入: 1->2->3->4->5->NULL

输出: 1->3->5->2->4->NULL

示例 2:

输入: 2->1->3->5->6->4->7->NULL

输出: 2->3->6->7->1->5->4->NULL

说明:

应当保持奇数节点和偶数节点的相对顺序。
链表的第一个节点视为奇数节点,第二个节点视为偶数节点,以此类推。

思路

之前说过,增删节点是链表的重要基础技巧,本道题就体现的很深刻。

从题意得知,奇数节点都在前面,偶数节点都在后面,即把1->2->3->4->5->NULL变成1->3->5->2->4->NULL,如下图所示:


可以看到问题的关键是奇数节点和偶数节点交替抽出成两条独立的链表,最终再合成一条新的链表。

代码
/**
 * Definition for singly-linked list.
 * function ListNode(val) {
 *     this.val = val;
 *     this.next = null;
 * }
 */
/**
 * @param {ListNode} head
 * @return {ListNode}
 */
var oddEvenList = function(head) {
    if(head === null || head.next === null || head.next.next === null)
        return head;
    
    let cur1 = head, cur2 = head.next;
    let evenHead = head.next;
    while(cur1.next && cur2.next) {
        cur1.next = cur2.next;
        cur1 = cur1.next;
        cur2.next = cur1.next;
        cur2 = cur2.next;
    }
    cur1.next = evenHead;
    
    return head;
};

欢迎关注前端亚古兽(fe-yagushou),更多前端以及互联网周边知识推送。


耳东
766 声望51 粉丝

知乎专栏:[链接]