每日两道力扣,day6

每日两道力扣,day6

每日两道力扣,day6

每日两道力扣,今天是:

11. 盛最多水的容器 - 力扣(LeetCode)

15. 三数之和 - 力扣(LeetCode)

第一题:盛最多水的容器

11. 盛最多水的容器 - 力扣(LeetCode)

1.思路:

在写这个题之前,咱们需要了解一个经典原理------木桶效应。

显然,在相同底面积的情况下,木桶盛水的最大值,由最短的那块板决定。这个题很明显是双指针算法的应用场景。因为这个题目给出的是一个平面切割图,咱们定义left,right左右两个指针。底面积S = right - left。高度应是min(heightleft,heightright),所以体积v就是这二者的乘积。观察题目给的示例图,当heightleft < heightright 时,left应该往右移动。反之,right应该往左移动。因为数组有限,我们只需要保证程序不会运行超时即可,剩下的交给计算机的算力。

2.代码实现:
复制代码
class Solution {
public:
    int maxArea(vector<int>& height) {
        int left = 0, right = height.size() - 1;
        int ret = 0;
        while(left < right)
        {
            int v = min(height[left], height[right]) * (right - left);
            ret = max(v,ret);
            if(height[left] < height[right])
            left++;
            else
            right--;
        }
        return ret;
    }
};
3.细节:

这个题需要理解木桶效应,以及示例图。较为简单,类似小学课本上的看图说话。

第二题:三数之和

15.三数之和-力扣leetcode

1.思路:

在写这道题的时候,我不知道大家有没有写过两数之和。有句话是这么说的:有人相爱,有人夜里看海,有人leetcode第一题要用两个for循环。小编就属于for循环大军里的一员,不过那个题我其实还会两种解法,排序+双指针,哈希(这个是看题解才知道的,我那会还没学到哈希呢)。

同样的这道三数之和,咱们也可以运用三个for循环暴力求解,但是我感觉99.99%会超时。3个for循环,时间复杂度是O(N3),你小子要是在面试的时候敢这样写,面试官分分钟让你进人才管理库。

还记得咱们昨天讲的双指针算法具有降维的特点吗?暴力法要用3个for循环,时间复杂度是O(N3),我们直接降为O (N2)。优雅,太优雅了。

具体落实的话是:

第一步,利用sort将数组排序

第二步,双指针算法里面定义一个target = -numsi

第三步,将numsleft 与numsrifght的和,同target比较大小,推进左右指针(然后得到一组解)

但是我按照这三步落实代码的时候,测试样例跑不满,说明我们的代码还是存在可以优化的地方。这个时候我们就得想一想,哪里是不够完美的,不够优雅的呢?

(1)是sort吗?当然不是啊,我们只是调了一个库函数sort帮我们将数组排序。

(2)那既然排好序了,所以大小关系肯定是numsi < numsleft <numsright咯,而我们在双指针算法里面定义一个target = -numsi,那当numsi > 0的时候,那必然是不存在解的,所以我们直接break就好了。

(3)在经历(2)的优化后,我们发现测试样例还是跑不满。会出现重复的多余答案。显然对于numsi而言,当i++后,numsi很可能和nusmi-1相等啊。这个逻辑bug刚好和咱们的实践吻合,因此咱们必须把这里给优化。在i++后,写一个while循环,当i < n && numsi == numsi-1,在防止数组越界的前提下,让i往后走一步。

2.代码实现:
复制代码
class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        vector<vector<int>> ret;
        sort(nums.begin(),nums.end());
        int n = nums.size();
        for(int i = 0; i < n; )
        {
            int target = -nums[i];
            int left = i+1;
            int right = n - 1;

            while(left < right)
            {
                //因为已经排完序了
                //优化1
                if(nums[i] > 0)
                break;

                if(nums[left] + nums[right] > target)
                {
                    right--;
                }
                else if(nums[left] + nums[right] < target)
                {
                    left++;
                }
                else
                {
                    ret.push_back({nums[i],nums[left],nums[right]});
                    left++;
                    right--;
                    //跑不满啊,不要忘记我们是已经排好序了的
                    //所以我又来优化了
                    //优化2.
                    while(left < right && nums[left] == nums[left-1])
                    left++;
                    while(left < right && nums[right] == nums[right+1])
                    right--;
                }
            }
            //还是跑不满,不要忘记我们是已经排好序了的
            //优化3.
            i++;
            while(i < n && nums[i] == nums[i-1])
            i++;

        }
        return ret;
    }
    
};
3.细节:

这个题的话不需要用到long long,但必须看懂思路里面的优化部分,不然就只能进人才管理库了。

先打个预防针,咱们明天要更新的是四数之和,接雨水。估计会很难,各位请做好心理准备。

好了,今天的每日两道力扣到这里就算是结束了,看完是不是感觉有所收获呢?如果学有所获的话,麻烦给个三连支持一下呗。感谢观看,您的支持,将是我前进路上的重要动力。

相关推荐
搬砖魁首36 分钟前
基础能力系列 - 多线程2 - 条件变量
c++·rust·条件变量·原子类型·线程同步互斥
youngerwang41 分钟前
【从搬运工到协处理器:网卡芯片架构、算法、验证与边缘演进深度剖析】
网络·算法·架构·芯片
chase_my_dream44 分钟前
C++ + SLAM 高频面试问题整理
开发语言·c++·面试
想要成为糕糕手1 小时前
前端必修课:JavaScript 数组与数据结构底层逻辑全解析
javascript·数据结构·面试
牛油果子哥q1 小时前
【C++ STL string 】C++ STL string 终极精讲:底层原理、内存机制、全套API、深浅拷贝、易错坑点与工程实战规范
数据库·c++
KaMeidebaby1 小时前
卡梅德生物技术快报|纯化重组蛋白实操详解
人工智能·python·tcp/ip·算法·机器学习
手写码匠2 小时前
从零实现 Prompt 工程引擎:结构化提示、自动优化与多轮自省体系
人工智能·深度学习·算法·aigc
无限码力2 小时前
阿里算法岗 0530笔试真题 - 多约束条件下的元素匹配统计
算法·阿里笔试真题·阿里机试真题·阿里算法岗笔试
lqqjuly2 小时前
MLA — 多头潜在注意力深度解析
深度学习·神经网络·算法
吴可可1233 小时前
SolidWorks草图转三维DWG技巧
算法