算法入门:专题一:双指针(有效三角形的个数)

简介: 给定一个数组,找出能组成三角形的三元组个数。利用“两边之和大于第三边”的性质,先排序,再用双指针优化。固定最大边,左右指针从区间两端向内移动,若两短边之和大于最长边,则中间所有组合均有效,时间复杂度由暴力的O(n³)降至O(n²)。

leetcode题目有效三角形个数
link
22217.png

题目解析:

数组中随意挑出三个数字,挑出所有可以组成三角形的组合,最终仅仅需要返回所有三元组的个数即可。

算法原理:

三角形的组成:两边之和大于第三边,两边之差小于第三边。那如果按照这种规则拿出所有的组合,再将所有组合进行判断,这个时间复杂度可能会很大,那么要减少时间的消耗,就需要先对三个数字进行排序,只需要较小的两个数字相加大于第三个数字就满足构成三角形的条件了。

优化先对整个数组进行排序

解法一:暴力解法,三层枚举

解法二:排序之后,利用单调性,使用双指针算法来解决问题。先来固定最大的数字,这里给一个数组我们好理解,[ 2, 2, 3, 4, 5, 9, 10],先固定第三边c=10,然后将定义两个指针,left先从索引为0的元素开始,假设left所在的数字数值为a,right是C之前的索引为n-2的元素,指向第二大的元素,right所在的数字数值为b,如果最小的元素数值跟第二大的元素数值相加大于最大的数值,那么此时2之后的元素跟9相加都会大于10,此时有right-left组可以构成a+b>c的三角形组合。
接下来将right--,向前挪动一位,如果a+b都是小于等于c的,找不到结果,然后left++,接着重复我们上面的操作,

解法总结:

1.先固定最大的数;
2.在最大的数的区间中,使用双指针算法,快速统计出符合要求的三元组的个数;

接下来我们将思路转化成代码:
先对整个数组 进行排序,sort(nums.begin(),nums.end());
整个题目最后让我们统计出符合条件的三元组的个数,所以定义一个ret来存储符合条件的三元组的个数,最后返回ret即可;
接着需要利用双指针来统计符合元素的三元组,在此之前需要有一个数组中大的数字充当c,但是这个C不是固定的,让c从数组最后一个元素向前遍历到索引为2的数字,for(int i=n-1;i>=2;i--)
接着让left指针向后遍历,right向前遍历,

  • a+b>c: 如果left指向的位置的数字加上right指向的位置的数字大于c,那么从left所在位置直到right所在位置之前这一段区间的数字加上right都大于c,都满足三元组的条件,ret+=right-left,right--;
  • a+b<=c:left指向位置的数字加right指向位置的数字小于c,说明left太小了,left++;继续进行判断。最终left>=right时结束循环。
class Solution {
   
public:
    int triangleNumber(vector<int>& nums) {
   
        //1.优化
        sort(nums.begin(),nums.end());
        //2.利用双指针解决算法问题
        int ret=0;//统计三元组
        int n=nums.size()-1;
        //固定一个最大的数字作为c
        for(int i=n-1;i>=2;i--)//先固定最大的数字
        {
   
            //利用双指针快速统计符合条件的三元组的个数
            int left=0,right=i-1;
            while(left<right)
            {
   
                if(nums[left]+nums[right]>nums[i]) ret+=right-left,right--;
                else left--;
            }

        }
        return ret;
    }
};
相关文章
|
2月前
|
存储 算法
算法入门:专题二---滑动窗口(长度最小的子数组)类型题目攻克!
给定一个正整数数组和目标值target,找出总和大于等于target的最短连续子数组长度。利用滑动窗口(双指针)优化,维护窗口内元素和,通过单调性避免重复枚举,时间复杂度O(n)。当窗口和满足条件时收缩左边界,更新最小长度,最终返回结果。
|
2月前
|
存储 算法 编译器
算法入门:剑指offer改编题目:查找总价格为目标值的两个商品
给定递增数组和目标值target,找出两数之和等于target的两个数字。利用双指针法,left从头、right从尾向中间逼近,根据和与target的大小关系调整指针,时间复杂度O(n),空间复杂度O(1)。找不到时返回{-1,-1}。
|
5月前
|
机器学习/深度学习 数据采集 算法
你天天听“数据挖掘”,可它到底在“挖”啥?——数据挖掘算法入门扫盲篇
你天天听“数据挖掘”,可它到底在“挖”啥?——数据挖掘算法入门扫盲篇
134 0
|
9月前
|
机器学习/深度学习 算法 机器人
强化学习:时间差分(TD)(SARSA算法和Q-Learning算法)(看不懂算我输专栏)——手把手教你入门强化学习(六)
本文介绍了时间差分法(TD)中的两种经典算法:SARSA和Q-Learning。二者均为无模型强化学习方法,通过与环境交互估算动作价值函数。SARSA是On-Policy算法,采用ε-greedy策略进行动作选择和评估;而Q-Learning为Off-Policy算法,评估时选取下一状态中估值最大的动作。相比动态规划和蒙特卡洛方法,TD算法结合了自举更新与样本更新的优势,实现边行动边学习。文章通过生动的例子解释了两者的差异,并提供了伪代码帮助理解。
742 2
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
417 7
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
算法 容器
【算法】——双指针算法合集(力扣)
移动零,复写零,快乐数,盛最多水的容器,有效三角形的个数,和为s的两个数(查找总价格为目标值的两个商品 ),三数之和,四数之和
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
1249 13
|
存储 人工智能 C语言
C语言程序设计核心详解 第八章 指针超详细讲解_指针变量_二维数组指针_指向字符串指针
本文详细讲解了C语言中的指针,包括指针变量的定义与引用、指向数组及字符串的指针变量等。首先介绍了指针变量的基本概念和定义格式,随后通过多个示例展示了如何使用指针变量来操作普通变量、数组和字符串。文章还深入探讨了指向函数的指针变量以及指针数组的概念,并解释了空指针的意义和使用场景。通过丰富的代码示例和图形化展示,帮助读者更好地理解和掌握C语言中的指针知识。
560 4
|
C语言
无头链表二级指针方式实现(C语言描述)
本文介绍了如何在C语言中使用二级指针实现无头链表,并提供了创建节点、插入、删除、查找、销毁链表等操作的函数实现,以及一个示例程序来演示这些操作。
195 0
|
编译器 C语言
【C语言初阶】指针篇—下
【C语言初阶】指针篇—下