当前位置:首页 > 力扣 > 力扣15题三数之和解法(C++双指针算法详解)

力扣15题三数之和解法(C++双指针算法详解)

7个月前 (07-30)

力扣15题三数之和解法(C++双指针算法详解) 力扣题解 三数之和 C++ 双指针 指针 第1张

一、题目解读

力扣15题(三数之和)要求在一个包含n个整数的数组中,找出所有三个数之和为0的组合,且每个组合的元素不能重复。题目考察数组遍历、排序算法双指针技巧的结合,是经典的多指针问题,对时间复杂度优化有较高要求。

二、解题思路

采用“双指针”策略:首先对原数组排序,然后固定第一个数,通过左右指针在剩余区间内寻找和为目标的数对。核心逻辑在于利用排序后的有序性,通过双指针移动减少无效遍历,并跳过重复元素避免结果重复。

三、解题步骤

1. 排序预处理:对输入数组进行升序排序,为后续双指针操作奠定基础。

2. 外层循环固定首元素:遍历数组,每次固定第一个数(需跳过重复值)。

3. 双指针搜索:

○ 初始化左指针为i+1,右指针为数组末尾;

○ 计算当前三数之和,若为0则记录结果,并移动左右指针时需跳过重复元素;

○ 若和小于0,左指针右移扩大和值;若和大于0,右指针左移缩小和值。

4. 循环终止条件:确保左指针始终在右指针左侧,避免重复计算。

四、代码与注释

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        vector<vector<int>> result;
        int n = nums.size();
        
        // 1. 排序预处理
        sort(nums.begin(), nums.end());
        
        // 2. 固定首元素遍历
        for (int i = 0; i < n - 2; ++i) {
            // 跳过重复首元素(优化)
            if (i > 0 && nums[i] == nums[i - 1]) continue;
            
            int left = i + 1;     // 左指针
            int right = n - 1;    // 右指针
            int target = -nums[i]; // 目标值(三数和为0,即其余两数和为-target)
            
            // 3. 双指针搜索
            while (left < right) {
                int sum = nums[left] + nums[right];
                if (sum == target) {
                    // 找到有效组合
                    result.push_back({nums[i], nums[left], nums[right]});
                    // 跳过重复的left和right元素
                    while (left < right && nums[left] == nums[left + 1]) ++left;
                    while (left < right && nums[right] == nums[right - 1]) --right;
                    ++left;  // 移动指针
                    --right;
                } else if (sum < target) {
                    ++left;   // 和太小,右移左指针
                } else {
                    --right;  // 和太大,左移右指针
                }
            }
        }
        return result;
    }
};

注释说明:代码中通过排序降低复杂度,双指针动态调整搜索范围,并利用跳过重复元素避免冗余计算,确保结果正确且不重复。

五、总结

该解法通过排序和双指针技术将时间复杂度优化至O(n^2),空间复杂度O(1)。关键在于利用有序数组的特性,通过首元素固定与双指针动态收缩区间,同时结合去重逻辑保证结果唯一性。适用于求解多元素和为定值的问题,是算法面试中的高频考点。



原创内容 转载请注明出处

分享给朋友:

相关文章

力扣198.打家劫舍|动态规划解法中的特殊边界处理

力扣198.打家劫舍|动态规划解法中的特殊边界处理

题意解析:在排列成直线的房屋群中,每个房屋藏有价值不同的财物。小偷不能连续抢劫相邻的两间房屋,否则会触发警报。我们需要设计一套抢劫策略,使得在不触发警报的前提下,能够获取的最大财物总和。这个问题本质上...

力扣5:中心扩散法 轻松破解最长回文子串

力扣5:中心扩散法 轻松破解最长回文子串

题目解读:在一个给定的字符串中,我们需要找到最长的回文子串。回文是指正读反读都相同的字符串,如"aba"、"abba"都是回文。这个问题看似简单,但要在字符串中...

IOI 1994 洛谷1216:如何用O(1)空间解决数字三角形问题?附代码实现

IOI 1994 洛谷1216:如何用O(1)空间解决数字三角形问题?附代码实现

题目重解:数字三角形是一个经典的动态规划问题,给定一个由数字组成的三角形结构,从顶部出发,每次可以移动到下方相邻的数字,最终到达底部。我们需要找到一条路径,使得路径上经过的数字总和最大。这个问题可以很...

力扣3112题解法:带时间限制的最短路径问题解析(C++代码)

力扣3112题解法:带时间限制的最短路径问题解析(C++代码)

一、题目解读力扣3112题要求解决带时间限制的最短路径问题:给定一个有向图,节点具有消失时间,需计算从起点到各节点的最短路径,且路径总时间不能超过节点的消失时间。题目难点在于需在传统最短路径算法(如D...

【力扣2846题】图论+二进制提升:高效解决连通性问题(附C++代码)

【力扣2846题】图论+二进制提升:高效解决连通性问题(附C++代码)

一、题目解读力扣2846题要求解决一个基于图连通性的操作优化问题。给定一个无向图,包含边权重,以及一系列查询,每个查询询问两点间路径的最小操作次数。题目关键在于高效计算路径上权重分布的统计信息,并转化...

洛谷1656题解:基于Tarjan算法求解割边问题(附代码与详细步骤)

洛谷1656题解:基于Tarjan算法求解割边问题(附代码与详细步骤)

一、题目解读洛谷1656题要求在无向图中找出所有割边(即删除后导致图不连通的边)。题目核心在于判断图的连通性,并识别哪些边是“桥”。需理解图论中的连通分量概念,以及如何通过算法高效定位割边。二、解题思...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。