LeetCode算法题(Go语言实现)_32

题目

在一个大小为 n 且 n 为 偶数 的链表中,对于 0 <= i <= (n / 2) - 1 的 i ,第 i 个节点(下标从 0 开始)的孪生节点为第 (n-1-i) 个节点 。

比方说,n = 4 那么节点 0 是节点 3 的孪生节点,节点 1 是节点 2 的孪生节点。这是长度为 n = 4 的链表中所有的孪生节点。

孪生和 定义为一个节点和它孪生节点两者值之和。

给你一个长度为偶数的链表的头节点 head ,请你返回链表的 最大孪生和 。

一、代码实现

go 复制代码
func pairSum(head *ListNode) int {
    // 快慢指针找中点
    slow, fast := head, head
    for fast.Next != nil && fast.Next.Next != nil {
        slow = slow.Next
        fast = fast.Next.Next
    }
    
    // 反转后半部分链表
    var prev *ListNode
    current := slow.Next
    slow.Next = nil // 断开前后半链表
    for current != nil {
        next := current.Next
        current.Next = prev
        prev = current
        current = next
    }
    
    // 计算最大孪生和
    maxSum := 0
    p1, p2 := head, prev
    for p1 != nil && p2 != nil {
        sum := p1.Val + p2.Val
        if sum > maxSum {
            maxSum = sum
        }
        p1 = p1.Next
        p2 = p2.Next
    }
    return maxSum
}

二、算法分析

1. 核心思路
  • 双指针分割:通过快慢指针将链表分为前后两半
  • 链表反转:反转后半部分链表以实现对称访问
  • 并行遍历:同时遍历前半段和反转后的后半段计算孪生和
2. 关键步骤
  1. 快慢指针定位中点:快指针走两步,慢指针走一步,最终慢指针指向前半段末尾
  2. 反转后半链表:采用经典三指针法反转链表,时间复杂度 O(n/2)
  3. 孪生和计算:通过双指针并行遍历,比较每个对称节点对的和值
3. 复杂度
指标 说明
时间复杂度 O(n) 遍历链表三次(分割、反转、求和)
空间复杂度 O(1) 仅使用固定数量的指针变量

三、图解示例

四、边界条件与扩展

1. 特殊场景验证
  • 最小偶长链表:输入链表[1,2] → 输出3(直接首尾相加)
  • 全相同元素:输入链表[5,5,5,5] → 输出10(每对和均为10)
  • 极值分布:输入链表[0,100000] → 输出100000(边界测试)
2. 多语言实现
python 复制代码
# Python实现
class Solution:
    def pairSum(self, head: ListNode) -> int:
        # 快慢指针分割
        slow = fast = head
        while fast.next and fast.next.next:
            slow = slow.next
            fast = fast.next.next
        
        # 反转后半部分
        prev, curr = None, slow.next
        slow.next = None
        while curr:
            next_node = curr.next
            curr.next = prev
            prev, curr = curr, next_node
        
        # 计算最大值
        max_sum = 0
        p1, p2 = head, prev
        while p1 and p2:
            max_sum = max(max_sum, p1.val + p2.val)
            p1, p2 = p1.next, p2.next
        return max_sum
java 复制代码
// Java实现
class Solution {
    public int pairSum(ListNode head) {
        // 快慢指针定位中点
        ListNode slow = head, fast = head;
        while (fast.next != null && fast.next.next != null) {
            slow = slow.next;
            fast = fast.next.next;
        }
        
        // 反转后半链表
        ListNode prev = null, curr = slow.next;
        slow.next = null;
        while (curr != null) {
            ListNode next = curr.next;
            curr.next = prev;
            prev = curr;
            curr = next;
        }
        
        // 遍历求最大值
        int maxSum = 0;
        ListNode p1 = head, p2 = prev;
        while (p1 != null && p2 != null) {
            maxSum = Math.max(maxSum, p1.val + p2.val);
            p1 = p1.next;
            p2 = p2.next;
        }
        return maxSum;
    }
}

五、总结与扩展

1. 核心创新点
  • 三段式处理:将问题分解为链表分割、反转、求和的独立模块
  • 空间优化:通过原地反转避免数组存储,实现O(1)空间复杂度
  • 对称访问机制:利用链表反转实现自然对称遍历
2. 扩展应用
  • 回文链表检测:结合快慢指针与反转操作
  • 环形链表处理:通过快慢指针检测环的存在
  • K个一组反转:将本解法中的反转逻辑扩展为分组处理
3. 工程优化方向
  • 内存预判:根据链表长度选择数组存储或原地反转策略
  • 并发安全:添加原子操作保证多线程环境下的指针安全
  • 缓存优化:利用内存局部性原理优化遍历顺序
相关推荐
慕芊妤11 分钟前
Logo语言的数据可视化
开发语言·后端·golang
十七算法实验室25 分钟前
Matlab实现鼠群优化算法优化随机森林算法模型 (ROS-RF)(附源码)
开发语言·算法·决策树·随机森林·机器学习·支持向量机·matlab
烁34729 分钟前
每日一题(小白)字符串娱乐篇16
java·开发语言·算法·娱乐·暴力
butiehua020231 分钟前
Go语言常用算法实现
数据结构·算法·golang·排序算法
luckycoding39 分钟前
1631. 最小体力消耗路径
数据结构·算法·leetcode
.YY001.43 分钟前
数据结构第一轮复习--第七章查找(包含课程代码)
数据结构·算法
LuckyLay1 小时前
LeetCode算法题(Go语言实现)_30
算法·leetcode·golang
对方正在长头发丿1 小时前
P1162 填涂颜色(BFS)
数据结构·c++·算法·蓝桥杯·宽度优先
MeyrlNotFound1 小时前
杂谈:抖音双塔召回模型:亿级内容分发的数学魔法
算法
不爱学英文的码字机器1 小时前
隐私计算的崛起:数据安全的未来守护者
服务器·算法