力扣-链表最大孪生和

思路分析

  1. 找链表中点:用快慢指针(慢指针走 1 步,快指针走 2 步)找到链表的中间节点;
  2. 反转后半段链表:将中点后的后半段链表反转;
  3. 计算孪生和:用两个指针分别从链表头部和反转后的后半段头部出发,依次计算每对孪生节点的和,记录最大值。

代码实现

java 复制代码
/**
* 方法一:快慢指针找到中间节点,反转后半部分链表,遍历两部分节点计算最大和
* @param head
* @return
*/
public int pairSum(ListNode head) {
   // 快慢指针找到中间节点
   ListNode slow = head;
   ListNode fast = head;
   while (fast != null && fast.next != null) {
       slow = slow.next;
       fast = fast.next.next;
   }
   // 反转后半部分链表
   ListNode reverseHead = reverseListNode(slow);
   // 遍历两部分节点计算最大和
   int maxSum = 0;
   while (reverseHead != null) {
       int firstNum = head.val;
       int secondNum = reverseHead.val;
       maxSum = Math.max(maxSum, firstNum + secondNum);
       head = head.next;
       reverseHead = reverseHead.next;
   }
   return maxSum;
}

/**
* @Author Feng
* @Description
* @Date 2026/1/14
* @Param [slow]
* @return main.leetcode75.arr_str.entity.ListNode
**/
private ListNode reverseListNode(ListNode head) {
   // 定义前一个节点为null,当前节点为头节点
   ListNode prev = null;
   ListNode curr = head;
   while (curr != null) {
       ListNode temp = curr.next;
       curr.next = prev;
       prev = curr;
       curr = temp;
   }
   return prev;
}

复杂度分析

  • 空间复杂度 O (1):无需额外存储所有节点值,仅用指针操作;
  • 时间复杂度 O (n):找中点 O (n/2) + 反转后半段 O (n/2) + 计算和 O (n/2),总复杂度 O (n)。

思路分析二

  1. 使用快慢指针找中点:

    • 使用快慢指针技术找到链表的中间节点
    • 慢指针每次移动一步,快指针每次移动两步
    • 当快指针到达末尾时,慢指针正好在链表的中点位置
  2. 利用栈存储前半部分节点值:

    • 创建一个双端队列(用作栈)
    • 从头节点开始遍历到中点之前的所有节点
    • 将这些节点的值依次压入栈中
    • 由于栈是后进先出的数据结构,这样栈顶元素对应的是链表后半部分对称位置的节点
  3. 配对求最大和:

    • 从中点开始遍历后半部分链表
    • 每次从栈中弹出一个值(这对应前半部分对称位置的节点值)
    • 将栈中弹出的值与当前后半部分节点的值相加
    • 更新最大和

代码实现二

java 复制代码
public int pairSum2(ListNode head) {
    // 快慢指针找到中间节点
    ListNode slow = head;
    ListNode fast = head;
    while (fast != null && fast.next != null) {
        slow = slow.next;
        fast = fast.next.next;
    }
    // 遍历前半部分节点,将节点值加入栈中
    Deque<Integer> stack = new ArrayDeque<>();
    while (head != slow) {
        stack.push(head.val);
        head = head.next;
    }
    // 遍历后半部分节点,弹出栈顶元素与当前节点值计算最大和
    int maxSum = 0;
    while (slow != null) {
        int firstNum = stack.pop();
        int secondNum = slow.val;
        maxSum = Math.max(maxSum, firstNum + secondNum);
        slow = slow.next;
    }
    return maxSum;
}

复杂度分析

  • 时间复杂度:O(n),只需要遍历链表两次
  • 空间复杂度:O(n/2),只需要额外的空间存储前半部分节点的值
    优势: 不需要修改原链表结构
相关推荐
We་ct3 分钟前
LeetCode 228. 汇总区间:解题思路+代码详解
前端·算法·leetcode·typescript
AIpanda8887 分钟前
如何借助AI销冠系统提升数字员工在销售中的成效?
算法
啊阿狸不会拉杆8 分钟前
《机器学习导论》第 7 章-聚类
数据结构·人工智能·python·算法·机器学习·数据挖掘·聚类
木非哲13 分钟前
机器学习--从“三个臭皮匠”到 XGBoost:揭秘 Boosting 算法的“填坑”艺术
算法·机器学习·boosting
Re.不晚17 分钟前
JAVA进阶之路——数据结构之线性表(顺序表、链表)
java·数据结构·链表
小辉同志20 分钟前
437. 路径总和 III
算法·深度优先·广度优先
笨笨阿库娅25 分钟前
从零开始的算法基础学习
学习·算法
不想睡觉_32 分钟前
优先队列priority_queue
c++·算法
那个村的李富贵9 小时前
CANN加速下的AIGC“即时翻译”:AI语音克隆与实时变声实战
人工智能·算法·aigc·cann
power 雀儿9 小时前
Scaled Dot-Product Attention 分数计算 C++
算法