LeetCode 160. 相交链表 | 三种解法吃透核心逻辑(哈希表 + 双指针 + 长度对齐)

前言

相交链表是链表类面试的高频题(难度★★☆☆☆),核心考察对链表遍历、指针操作的理解。最初我用哈希表解决,后续又学习了官方双指针法,以及更易理解的「长度对齐法」,三种解法各有优劣,现将完整思路、避坑点和复杂度分析整理如下。

题目描述

160. 相交链表 给你两个单链表的头节点 headAheadB,请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点,返回 null

++注:相交的定义是「节点地址相同」,而非节点值相同。++

解法一:哈希集合(直观易想,空间换时间)

核心思路

利用哈希集合存储其中一个链表的所有节点,再遍历另一个链表,第一个在集合中匹配到的节点就是相交节点;若遍历完未匹配,说明无交点。

实现代码

cpp 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {
        unordered_set<ListNode*> nodeSet;
        // 存储链表A的所有节点
        while(headA != NULL){
            nodeSet.insert(headA);
            headA = headA->next;
        }
        // 遍历链表B,查找相交节点
        while(headB != NULL){
            if(nodeSet.find(headB) != nodeSet.end()){
                return headB;
            }
            headB = headB->next;
        }
        return NULL;
    }
};

复杂度分析

m、n分别为两个链表的长度

时间复杂度:O(m+n):遍历链表 A 耗时 O(m),遍历链表 B 匹配耗时 O(n);

空间复杂度:O(m),需存储链表 A 的所有节点,空间开销较大。

解法二:官方双指针法(最优解)

核心思路

两个指针分别从 headAheadB 出发,遍历到链表末尾时跳转到另一个链表的头节点继续遍历。

  • 若链表相交:两个指针会在相交节点处相遇(移动总距离均为 m+n);
  • 若链表不相交:两个指针最终都会指向 NULL(仍满足 p==q,直接返回即可)。

实现代码(避坑版)

cpp 复制代码
class Solution {
public:
    ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {
        if(headA == NULL || headB == NULL) return NULL;
        ListNode* p = headA;
        ListNode* q = headB;
        // 核心:仅判断p!=q(允许p/q为NULL)
        while(p != q){
            // 指针到末尾则跳转到另一链表头,否则走next
            p = p == NULL ? headB : p->next;
            q = q == NULL ? headA : q->next;
        }
        // 最终p==q:要么是相交节点,要么都是NULL
        return p;
    }
};

避坑要点

重做这道题时曾写出「超时代码」,核心错误:

复制代码
// 错误代码核心问题
while (p != NULL && q != NULL && p != q) { // 多余的非空判断
    p = p->next != NULL ? p->next : headB; // 指针永远到不了NULL
    q = q->next != NULL ? q->next : headA;
}
  • 循环条件加 p!=NULL && q!=NULL:无交点时指针永远无法相等,陷入死循环;
  • 指针移动逻辑错误:p->next!=NULL 导致指针永远到不了链表末尾,无法跳转到另一链表,违背双指针核心逻辑。

复杂度分析

m、n分别为两个链表的长度

时间复杂度:O(m+n),最坏情况遍历两个链表所有节点;

空间复杂度:O(1),仅使用两个指针变量,常数级开销。

解法三:长度对齐法

核心思路

相交链表的「交点后长度」完全一致,因此先计算两个链表的长度差,让较长链表的指针先移动「长度差」步,使两个指针到交点的距离一致,再同步遍历,第一个相等的节点即为交点。

实现代码

cpp 复制代码
// 辅助函数:获取链表长度
int GetLength(ListNode* head) {
    int len = 0;
    while(head != NULL) {
        len++;
        head = head->next;
    }
    return len;
}

ListNode* getIntersectionNode(ListNode* headA, ListNode* headB) {
    if (headA == NULL || headB == NULL) return NULL;
    
    int lenA = GetLength(headA);
    int lenB = GetLength(headB);
    ListNode* p = headA;
    ListNode* q = headB;
    
    // 较长链表的指针先移动长度差
    while (lenA > lenB) {
        p = p->next;
        lenA--;
    }
    while (lenB > lenA) {
        q = q->next;
        lenB--;
    }
    
    // 同步遍历,找第一个相等的节点
    while (p != NULL && q != NULL) {
        if (p == q) return p;
        p = p->next;
        q = q->next;
    }
    return NULL;
}

复杂度分析

m、n分别为两个链表的长度

时间复杂度:O(m+n),计算长度遍历两次链表,对齐后遍历一次最短链表;

空间复杂度:O(1),仅使用长度变量和指针。

三种解法对比

解法 时间复杂度 空间复杂度 优点 缺点
哈希集合 O(m+n) O(m) 逻辑直观,易实现 空间开销大
官方双指针 O(m+n) O(1) 空间最优,代码极简 逻辑较抽象,易踩死循环坑
长度对齐法 O(m+n) O(1) 逻辑易懂,面试易口述 需额外计算链表长度

核心总结

  1. 相交链表的核心是「节点地址相等」,而非值相等;
  2. 最优解是官方双指针法,需牢记「指针到末尾跳另一链表头 + 仅判断 p!=q」的核心逻辑;
  3. 长度对齐法更适合面试口述,哈希表法则适合快速实现,可根据场景选择。
相关推荐
不想看见4041 小时前
Maximal Square 基本动态规划:二维--力扣101算法题解笔记
算法·leetcode·动态规划
Hag_201 小时前
LeetCode Hot100 53.最大子数组和
数据结构·算法·leetcode
pursuit_csdn2 小时前
LeetCode 1461. Check If a String Contains All Binary Codes of Size K
算法·leetcode·职场和发展
Crazy________3 小时前
力扣113个mysql简单题解析(包含plus题目)
mysql·算法·leetcode·职场和发展
初次攀爬者3 小时前
力扣解题-无重复字符的最长子串
后端·算法·leetcode
圣保罗的大教堂4 小时前
leetcode 1461. 检查一个字符串是否包含所有长度为 K 的二进制子串 中等
leetcode
Trouvaille ~5 小时前
【动态规划篇】专题(二):路径问题——在网格图中的决策艺术
c++·算法·leetcode·青少年编程·动态规划
圣保罗的大教堂6 小时前
leetcode 761. 特殊的二进制字符串 困难
leetcode
Trouvaille ~6 小时前
【动态规划篇】专题(一):斐波那契模型——从数学递推到算法思维
c++·算法·leetcode·青少年编程·面试·动态规划·入门