题目: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. 动态扩容 → 处理不确定的返回数组大小


相关推荐
汀、人工智能18 小时前
[特殊字符] 第21课:最长有效括号
数据结构·算法·数据库架构·图论·bfs·最长有效括号
Boop_wu18 小时前
[Java 算法] 字符串
linux·运维·服务器·数据结构·算法·leetcode
故事和你9119 小时前
洛谷-算法1-2-排序2
开发语言·数据结构·c++·算法·动态规划·图论
Fcy64819 小时前
算法基础详解(三)前缀和与差分算法
算法·前缀和·差分
kvo7f2JTy19 小时前
基于机器学习算法的web入侵检测系统设计与实现
前端·算法·机器学习
List<String> error_P20 小时前
蓝桥杯最后几天冲刺:暴力大法(一)
算法·职场和发展·蓝桥杯
Hou'20 小时前
从0到1的C语言传奇之路
c语言·开发语言
迈巴赫车主21 小时前
蓝桥杯3500阶乘求和java
java·开发语言·数据结构·职场和发展·蓝桥杯
流云鹤21 小时前
Codeforces Round 1090 (Div. 4)
c++·算法
wljy121 小时前
第十三届蓝桥杯大赛软件赛省赛C/C++ 大学 B 组(个人见解,已完结)
c语言·c++·算法·蓝桥杯·stl