《算法题讲解指南:优选算法-双指针》--05有效三角形的个数,06查找总价值为目标值的两个商品

🔥小叶-duck个人主页

❄️个人专栏《Data-Structure-Learning》

《C++入门到进阶&自我学习过程记录》《算法题讲解指南》--从优选到贪心

未择之路,不须回头
已择之路,纵是荆棘遍野,亦作花海遨游


目录

05.有效三角形的个数

题目链接:

题目描述:

题目示例:

算法思路:

C++代码演示:

算法总结及流程解析:

06.查找总价值为目标值的两个商品

题目链接:

题目描述:

题目示例:

算法思路:

C++代码演示:

算法总结及流程解析:

结束语


05.有效三角形的个数

题目链接:

611. 有效三角形的个数 - 力扣(LeetCode)

题目描述:

题目示例:

解法:(排序+双指针)

算法思路:

先将数组排序。

判断三角形的优化方法:

  • 如果能构成三角形,需要满足任意两边之和大于第三边。但是实际上只需要让较小的两条边之和大于第三边即可

根据【上述优化思想】我们可以固定一个【最长边】 ,然后在比这条边小的有序数组中找出一个二元组,使得这个二元组之和大于这个最长边。由于数组是有序 的,我们可以利用【对撞指针】来优化

设最长边枚举到 i 位置 ,区间 【left,right】 是 i 位置左边的区间(也就是比它小的区间):

1.如果 nums left + nums right > nums i ;

  • 说明**【left,right - 1】** 区间上的所有元素均可以与nums right 构成比 **nums i **大的二元组
  • 满足条件的有 right - left
  • 此时 right 位置的元素的所有情况相当于全部考虑完毕,right--,进入下一轮判断

2.如果 nums left + nums right <= nums i

  • 说明left 位置的元素是不可能与 **【left + 1,right】**位置上的元素构成满足条件的二元组
  • left 位置的元素可以舍去,**left++**进去下轮循环

C++代码演示:

cpp 复制代码
class Solution {
public:
    int triangleNumber(vector<int>& nums) 
    {
        sort(nums.begin(), nums.end());
        int c = nums.size() - 1;
        int ret = 0;
        while(c > 1)
        {
            int left = 0;
            int right = c - 1;
            while(left < right)
            {
                if(nums[left] + nums[right] > nums[c])
                {
                    ret += (right - left);
                    right--;
                }
                if(nums[left] + nums[right] <= nums[c])
                {
                    left++;
                }
            }
            c--;
        }
        return ret;
    }
};

算法总结及流程解析:

06.查找总价值为目标值的两个商品

题目链接:

LCR 179. 查找总价格为目标值的两个商品 - 力扣(LeetCode)

题目描述:

题目示例:

解法:(双指针-对撞指针)

算法思路:

注意到本题是升序的数组,因此可以用【对撞指针】优化时间复杂度。

算法流程:

1.初始化 left,right 分别指向数组的左右两端(这里不是我们理解的指针,而是数组的下标)

2.当 left < right 的时候,一直循环

2.1当 nums left + nums right == target 时,说明找到结果,记录结果,并且返回;

2.2当 nums left + nums right < target 时:

  • 对于 numsleft 而言,此时numsright 相当于是 numsleft 能碰到的最大值(需要注意数组是升序)。如果此时不符合要求,说明在这个数组里面,没有别的数符合 nums left 的要求了。因此,我们可以大胆舍去这个数。让left++,去比较下一组数据;
  • 那对于nums right 而言,由于此时两数之和是小于目标值的,nums right 还可以选择比 nums left 大的值继续努力达到目标值,因此right指针我们按兵不动;

3.当 nums left + nums right > target 时。同理我们可以舍去 nums right 。让 right-- ,继续比较下一组数据,而 left 指针不变(因为他还是可以去匹配比 nums right 更小的数)

C++代码演示:

cpp 复制代码
class Solution {
public:
    vector<int> twoSum(vector<int>& price, int target) 
    {
        vector<int> ret;
        int left = 0;
        int right = price.size() - 1;
        while(left < right)
        {
            if(price[left] + price[right] > target)
            {
                right--;
            }
            else if(price[left] + price[right] < target)
            {
                left++;
            }
            else{
                ret.push_back(price[left]);
                ret.push_back(price[right]);
                break;
            }
        }
        return ret;
    }
};

算法总结及流程解析:

结束语

到此,05有效三角形的个数 和 06查找总价值为目标值的两个商品 两道算法题就讲解完了。这两道题所用方法都是**基于双指针算法的高效解题方法。强调排序预处理和指针移动策略在优化算法中的关键作用。**希望大家能有所收获!

相关推荐
To_OC3 小时前
LC 994 腐烂的橘子:人人都说是 BFS 入门题,我却写了三遍才过
javascript·算法·leetcode
金銀銅鐵7 小时前
[Python] 扩展欧几里得算法
python·数学·算法
To_OC9 小时前
LC 200 岛屿数量:经典 DFS 入门题,我第一次写居然连方向都搞错了
javascript·算法·leetcode
郝学胜_神的一滴14 小时前
CMake 30:循环语法全解|foreach_while双循环精讲、迭代技巧与实战避坑指南
c++·cmake
To_OC1 天前
LC 128 最长连续序列:别上来就排序,O (n) 解法才是这题的灵魂
javascript·算法·leetcode
05Kevin2 天前
lk每日冒险题--数据结构6.27
算法
To_OC2 天前
从一次栈溢出报错说起,我把递归彻底扒明白了
javascript·算法·程序员
千纸鹤安安2 天前
千问Qwen-AgentWorld来了:一个语言模型搞定七大Agent场景,GPT-5.4都输了
算法