题目:15. 三数之和

问题描述

给你一个整数数组 nums,判断是否存在三元组 [nums[i], nums[j], nums[k]],满足:

  • i != j != k

  • nums[i] + nums[j] + nums[k] == 0

返回 所有和为 0 且不重复的三元组

示例 1:

复制代码
输入:nums = [-1,0,1,2,-1,-4]
输出:[[-1,-1,2],[-1,0,1]]

示例 2:

复制代码
输入:nums = [0,1,1]
输出:[]

示例 3:

复制代码
输入:nums = [0,0,0]
输出:[[0,0,0]]

提示:

  • 3 <= nums.length <= 3000

  • -10^5 <= nums[i] <= 10^5


解题思路

  1. 排序 + 双指针

    • 先对数组进行升序排序,这样方便去重,同时可以用双指针快速找到两数之和。
  2. 固定第一个数

    • 遍历数组,固定 nums[i],剩下部分用双指针寻找两个数,使三数之和为 0。
  3. 双指针

    • 左指针 left = i + 1,右指针 right = n - 1

    • 计算 sum = nums[i] + nums[left] + nums[right]

      • sum == 0 → 找到答案,记录并去重

      • sum < 0 → 左指针右移

      • sum > 0 → 右指针左移

  4. 去重技巧

    • 遍历 i 时跳过重复元素

    • 左右指针移动时跳过重复元素

  5. 动态扩容结果数组

    • 因为满足条件的三元组数量不确定,需要动态扩容结果数组,防止越界。

C语言代码实现

复制代码
#include <stdlib.h>
#include <stdio.h>

// 数组排序函数
int cmp(const void* a, const void* b) {
    return (*(int*)a - *(int*)b);
}

int** threeSum(int* nums, int numsSize, int* returnSize, int** returnColumnSizes) {
    *returnSize = 0;
    if (numsSize < 3) {
        *returnColumnSizes = NULL;
        return NULL;
    }

    qsort(nums, numsSize, sizeof(int), cmp);  // 排序

    int capacity = 16;  // 初始容量
    int** result = malloc(sizeof(int*) * capacity);
    *returnColumnSizes = malloc(sizeof(int) * capacity);

    for (int i = 0; i < numsSize - 2; i++) {
        if (i > 0 && nums[i] == nums[i - 1]) continue;  // 去重

        int left = i + 1;
        int right = numsSize - 1;

        while (left < right) {
            int sum = nums[i] + nums[left] + nums[right];

            if (sum == 0) {
                if (*returnSize == capacity) {  // 扩容
                    capacity *= 2;
                    result = realloc(result, sizeof(int*) * capacity);
                    *returnColumnSizes = realloc(*returnColumnSizes, sizeof(int) * capacity);
                }

                result[*returnSize] = malloc(sizeof(int) * 3);
                result[*returnSize][0] = nums[i];
                result[*returnSize][1] = nums[left];
                result[*returnSize][2] = nums[right];
                (*returnColumnSizes)[*returnSize] = 3;
                (*returnSize)++;

                while (left < right && nums[left] == nums[left + 1]) left++;   // 去重
                while (left < right && nums[right] == nums[right - 1]) right--; // 去重
                left++;
                right--;
            }
            else if (sum < 0) left++;
            else right--;
        }
    }

    return result;
}

算法复杂度分析

项目 复杂度
排序 O(n log n)
双指针遍历 O(n²)
总时间复杂度 O(n²)
空间复杂度 O(1)(不算返回结果数组)

核心总结

  1. 排序 → 便于去重和双指针搜索

  2. 双指针 → 高效寻找两数之和

  3. 去重 → 防止重复三元组

  4. 动态扩容 → 处理不确定的返回数组大小


相关推荐
生信研究猿13 分钟前
leetcode 416. 分割等和子集
算法·leetcode·职场和发展
hnjzsyjyj13 分钟前
洛谷 B3622:枚举子集(递归实现指数型枚举)← DFS
数据结构·dfs
狗哥哥20 分钟前
面包屑自动推导的算法设计:从“最短路径匹配”到工程可落地
算法·架构
聆风吟º1 小时前
【C标准库】深入理解C语言strcat函数:字符串拼接的利器
c语言·开发语言·strcat·库函数
6Hzlia1 小时前
【Hot 100 刷题计划】 LeetCode 24. 两两交换链表中的节点 | C++ 精准指针舞步
c++·leetcode·链表
隔壁大炮1 小时前
Day07-RNN介绍
人工智能·pytorch·rnn·深度学习·神经网络·算法·numpy
WL_Aurora2 小时前
Python 算法基础篇之什么是算法
python·算法
墨染天姬2 小时前
[AI]DeepSeek-R1的GRPO算法
人工智能·算法·php
D_C_tyu2 小时前
JavaScript | 数独游戏核心算法实现
javascript·算法·游戏