Leetcode刷题笔记3

news2025/1/18 7:19:26

18. 四数之和

18. 四数之和 - 力扣(LeetCode)

给你一个由 n 个整数组成的数组 nums ,和一个目标值 target 。请你找出并返回满足下述全部条件且不重复的四元组 [nums[a], nums[b], nums[c], nums[d]]
(若两个四元组元素一一对应,则认为两个四元组重复):

0 <= a, b, c, d < n
a、b、c 和 d 互不相同
nums[a] + nums[b] + nums[c] + nums[d] == target
你可以按 任意顺序 返回答案 。

示例 1:

输入:nums = [1,0,-1,0,-2,2], target = 0
输出:[[-2,-1,1,2],[-2,0,0,2],[-1,0,0,1]]

示例 2:

输入:nums = [2,2,2,2,2], target = 8
输出:[[2,2,2,2]]

解法一:排序 + 暴力枚举 + 理由set去重

超时

解法二:排序 + 双指针

[-2, -1, 0, 0, 1, 2] target
       [                 ] target - a
  a
       b [              ] target - a - b
        left  right
1. 先依次固定一个a;
2. 在a后面的区间内,利用“三数之和”找到三个数,使这三个数的和等于target - a即可

三数之和的大体思路:
1. 依次固定一个b;
2. 在b后面的区间内,利用“双指针”找到两个数,使这两个数的和等于target - a - b即可

处理细节问题:
1. 不重
去重a,b,left,right

2. 不漏

时间复杂度:O(N^3)

代码:C++

class Solution {
public:
    vector<vector<int>> fourSum(vector<int>& nums, int target) {
        vector<vector<int>> ret;
        // 排序
        sort(nums.begin(), nums.end());
        // 利用双指针解决问题
        int n = nums.size();
        for(int i=0; i<n; ) //固定a
        {
            // 利用三数之和
            for(int j=i+1; j<n; ) //固定b
            {
                // 双指针
                int left = j+1, right = n-1;
                long long aim = (long long)target - nums[i] - nums[j];
                while(left<right)
                {
                    int sum = nums[left] + nums[right];
                    if(sum < aim) left++;
                    else if(sum > aim) right--;
                    else
                    {
                        ret.push_back({nums[i], nums[j], nums[left++], nums[right--]});
                        // 去重1
                        while(left < right && nums[left] == nums[left - 1]) left++; //当left右移完之后和左边这个数相比,如果相同,说明重复元素,需要跳过
                        while(left < right && nums[right] == nums[right + 1]) right--;
                    }
                }
                // 去重2
                j++;
                while(j < n && nums[j] == nums[j - 1]) j++;
            }
            // 去重3
            i++;
            while(i < n && nums[i] == nums[i - 1]) i++;
        }
        return ret;
    }
};

209. 长度最小的子数组

209. 长度最小的子数组 - 力扣(LeetCode)

 

解法一:暴力枚举出所有子数组的和

时间复杂度:O(N^3)

 R 
[2, 3, 1, 2, 4, 3]
 L
先定义一个sum,计算左区间的和
比如:sum + 2 + 3 + ...
这样可以省去再遍历一遍数组

     R
[2, 3, 1, 2, 4, 3]
 L
sum = 5

         R
[2, 3, 1, 2, 4, 3]
 L
sum = 6
len 2 -> 3

...

                 R
[2, 3, 1, 2, 4, 3]
 L
sum = 12
len = 5

这个len是一直增长的,所以肯定不是最短的结果

接下来让L向左移动一位,然后继续让R从左开始移动
那这个时候从3开始的区间可以在上一个结果中找到

解法二:利用单调性,使用“同向双指针”来优化,也就是 滑动窗口

当使用暴力解法时。两个指针都可以做到不回退,都是向一个方向移动时就可以使用滑动窗口

1. 初始化;left = 0, right = 0

2. 进窗口

3. 判断是否是结果,然后更新结果(长度),再出窗口,判断len

2和3一直循环

以下是图解:

 

 
4. 为什么不往后枚举呢?因为已经知道接下来的情况再枚举也是白枚举,因为是正整数数组
利用单调性,规避了很多没有必要的枚举行为

5. 时间复杂度
进窗口要一个循环,判断也是一个循环,等于是两层循环套在一起
但是总的操作次数只是2n次
所以最终的时间复杂度是一个O(N)

代码:C++

class Solution {
public:
    int minSubArrayLen(int target, vector<int>& nums) {
        int n = nums.size();
        int sum = 0, len = INT_MAX; //如果定义0,那最终的求min结果就是0,所以定义一个大的变量,不干扰最终结果
        for(int left = 0, right = 0; right < n; right++)
        {
            // 进入窗口
            sum += nums[right];

            //判断要不要缩小窗口
            while(sum >= target)
            {
                len = min(len, right - left + 1); // 更新结果
                sum -= nums[left++]; // 出窗口
            }
            // 一直更新到窗口不符合要求位置,然后right++
        }
        return len == INT_MAX ? 0 : len; // 如果没有最终结果就返回0,不然就返回len
    }
};

3. 无重复字符的最长子串

3. 无重复字符的最长子串 - 力扣(LeetCode)

子串和字数组都是连续的一段


解法一:暴力枚举 + 哈希表(判断字符是否重复出现)

可以借助哈希表来判断是否有重复字符
遍历的时候把字符保存到哈希表里,当遍历到重复元素时,停止这个操作

 R
[d, e, a, b, c, a, b, c, a]
 L

                    R
[d, e, a, b, c, a, b, c, a]
 L

当a重复时,停止这个操作,然后移动L,

                     R
[d, e, a, b, c, a, b, c, a]
     L

时间复杂度:O(N^2)

解法二:利用规律,使用“滑动窗口”来解决问题

 R
[d, e, a, b, c, a, b, c, a]
 L

                      R
[[d, e, a, b, c], a, b, c, a]
  L

当a重复时,停止这个操作,然后移动L,

                       R
[[d, e, a, b, c], a, b, c, a]
              L

当发现区间里面有重复字符时,可以让L先跳过这个重复字符
R也不用回到L的位置,让R继续移动即可
此时就可以使用滑动窗口解决问题

1. 先定义L = 0和R = 0

2. 进窗口 -> 让字符进入哈希表

3. 判断
根据判断结果判定是否出窗口(窗口内出现重复字符时才出窗口)

判断和出窗口是一个循环,一直到没有重复字符为止

出窗口就是从哈希表中删除该字符

然后更新结果(更新结果的过程要根据题目决定在哪)
本题中,在整个判断之后,更新结果

为什么要用滑动窗口,因为两个指针都不会回退

时间复杂度:O(N)
空间复杂度:O(1)
因为哈希表只有128位,所以可以忽略不计

代码:C++

class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        // 下标表示字符,让里面的数来表示是否重复出现,出现一次为1,两次为2...
        int hash[128] = {0}; // 使用数组表示哈希表

        int left = 0, right = 0, n = s.size();
        int ret = 0;
        while(right < n)
        {
            hash[s[right]]++; // 字符对应的下标,进入窗口

            while(hash[s[right]] > 1) // 判断
            {
                hash[s[left++]]--; // 先把哈希表里面的对应的值--,表示它从哈希表删除,然后++完成出窗口的操作
            }
            ret = max(ret, right - left + 1); // 更新结果,区间的长度是right - left + 1
            right++; // 让下一个元素进入窗口
        }
        return ret;
    }
};

1004. 最大连续1的个数 III

1004. 最大连续1的个数 III - 力扣(LeetCode)

最多k个0说明可以翻转0,1,2,...,一直到k个为止
比如下方这个例子中,k最多可以翻转100个0,但其实不需要翻转100次
[1,1,0,0,1,1,0], k = 100

[1,1,1,0,0,0,1,1,1,1,0], K = 2
               [                 ],最长为6

可以等价处理,在数组中满足0的个数不超过k次即可,只要这个区域的0不超过k,那么这个区域是一定可以翻转成功的
把原始问题转换成:
找出最长的子数组,0的个数不超过k个
 

解法一:暴力枚举 + zero计数器(int类型变量,统计0出现多少次)

 R
[1, 1, 1, 0, 0, 0, 1, 1, 1, 1, 0], K = 2
 L
zero = 0
             R
[1, 1, 1, 0, 0, 0, 1, 1, 1, 1, 0], K = 2
 L
zero = 1

                       R
[[1, 1, 1, 0, 0], 0, 1, 1, 1, 1, 0], K = 2 固定第一个位置的最优解
 L
zero = 2

然后R继续从数组最开始的位置开始移动

解法二:滑动窗口

所以可以让R越过这段区域,不用重新遍历,可以移动L
                     R
[1, 1, 1, 0, 0, 0, 1, 1, 1, 1, 0], K = 2 
                 L

当在暴力枚举中L和R都只往一个方向移动时,可以使用滑动窗口

1. left = 0, right = 0

2. 进窗口 -> 如果是1,无视;如果是0,计数器+1
当R向后移动时就相当于进窗口了
当进入窗口时无视,当碰到0时,计数器加一

3. 判断 -> 当zero大于k,出窗口
出窗口
让L一直右移,直到窗口合法为止

当判断结束之后,更新结果,当R移动到最后位置就会得到最终结果

时间复杂度:O(N)
空间复杂度:O(1)

代码:C++

class Solution {
public:
    int longestOnes(vector<int>& nums, int k) 
    {
        int ret = 0;
        for(int left = 0, right = 0, zero = 0; right < nums.size(); right++)
        {

            if(nums[right] == 0) zero++; // 进窗口
            while(zero > k) // 判断窗口是否合法
            {
                if(nums[left++] == 0) zero--; // 出窗口,left向后移动
            } 
            // 左闭右闭的区间
            ret = max(ret, right - left + 1); // 更新结果
        }
        return ret;
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1690602.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

骨位深间距小模具镶件如何走水路?3D打印让一切简单

在模具制造领域&#xff0c;骨位深且间距小的模具镶件由于结构复杂&#xff0c;传统加工方法难以制造出符合要求的冷却水路&#xff0c;导致模具在注塑过程中容易产生热量积聚&#xff0c;进而引发烫伤、缩孔等不良。然而&#xff0c;随着3D打印技术的飞速发展&#xff0c;这些…

爬虫逆向实例小记——某数据知识管理网站-DES-ECB模式

aHR0cHM6Ly9rZC5uc2ZjLmNuL2ZpbmFsUHJvamVjdEluaXQ 注意&#xff1a;本文是逆向部分比较少&#xff0c;主要为了流程走通&#xff0c;限于代码搬运工。 第一步:分析页面 此网站经过请求响应&#xff0c;可以看出响应内容为加密内容。 第二步&#xff1a;判断加密类型 在XHR …

【Linux】从零开始认识进程间通信 —— 管道

送给大家一句话&#xff1a; 人要成长&#xff0c;必有原因&#xff0c;背后的努力与积累一定数倍于普通人。所以&#xff0c;关键还在于自己。 – 杨绛 从零开始认识进程间通信 1 为什么要进程间通信2 进程如何通信3 进程通信的常见方式4 管道4.1 什么是管道4.2 管道通信的系…

postgresql insert on conflict 不存在则插入,存在则更新

向一张表执行插入动作&#xff0c;如果插入的字段数据已存在&#xff0c;则执行更新操作&#xff0c;不存在则进行插入操作。 1、创建一张表 CREATE TABLE "user_info" ( "id" int2 NOT NULL, "name" varchar(20) COLLATE "pg_catalog&quo…

近邻算法详细

近邻算法&#xff0c;特别是K-近邻算法&#xff08;K-Nearest Neighbors, KNN&#xff09;&#xff0c;是一种基于实例的学习方法&#xff0c;广泛应用于分类和回归分析任务。下面是K-近邻算法的详细说明&#xff1a; 基本概念 K-近邻算法的核心思想是“物以类聚”&#xff0…

基于transformers框架实践Bert系列1--分类器(情感分类)

本系列用于Bert模型实践实际场景&#xff0c;分别包括分类器、命名实体识别、选择题、文本摘要等等。&#xff08;关于Bert的结构和详细这里就不做讲解&#xff0c;但了解Bert的基本结构是做实践的基础&#xff0c;因此看本系列之前&#xff0c;最好了解一下transformers和Bert…

node版本管理nvm详细教程

安装 nvm 之前先清理node相关的所有配置&#xff0c;如环境变量、.npmrc文件、node_cache、node_global 等 一、下载nvm 任选一处下载即可 官网&#xff1a;Releases coreybutler/nvm-windows (github.com) 码云&#xff1a;nvm下载仓库: nvm下载仓库 百度网盘&#xff1…

基于GA遗传优化的CNN-GRU的时间序列回归预测matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 4.1 CNN-GRU模型架构 4.2 GA优化CNN-GRU流程 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 MATLAB2022a 3.部分核心程序 ...........................................…

雷电预警监控系统:守护安全的重要防线

TH-LD1在自然界中&#xff0c;雷电是一种常见而强大的自然现象。它既有震撼人心的壮观景象&#xff0c;又潜藏着巨大的安全风险。为了有效应对雷电带来的威胁&#xff0c;雷电预警监控系统应运而生&#xff0c;成为现代社会中不可或缺的安全防护工具。 雷电预警监控系统的基本…

代码随想录算法训练营第14天 |● 理论基础 ● 递归遍历 ● 迭代遍历 ● 统一迭代

文章目录 前言二叉树的递归遍历&#x1f496;递归算法基本要素代码 迭代遍历-需要先理清思路再写前向迭代法后序迭代中序迭代 迭代法统一写法总结 前言 理论基础 需要了解 二叉树的种类&#xff0c;存储方式&#xff0c;遍历方式 以及二叉树的定义 记录我容易忘记的点 题目…

打造AI虚拟伴侣 - 优化方案

第一部分:框架优化概述 1、精确定位: 构建一个高度灵活且用户友好的平台,旨在通过无缝集成多种大型语言模型(LLMs)后端,为用户创造沉浸式的角色交互体验。不仅适配电脑端,还特别优化移动端体验,满足二次元AI虚拟伴侣市场的特定需求。 2、核心功能强化: 增强后端兼容…

大数据Hive中的UDF:自定义数据处理的利器(下)

在上一篇文章中&#xff0c;我们对第一种用户定义函数&#xff08;UDF&#xff09;进行了基础介绍。接下来&#xff0c;本文将带您深入了解剩余的两种UDF函数类型。 文章目录 1. UDAF1.1 简单UDAF1.2 通用UDAF 2. UDTF3. 总结 1. UDAF 1.1 简单UDAF 第一种方式是 Simple(简单…

叶面积指数(LAI)数据、NPP数据、GPP数据、植被覆盖度数据获取

引言 多种卫星遥感数据反演叶面积指数&#xff08;LAI&#xff09;产品是地理遥感生态网推出的生态环境类数据产品之一。产品包括2000-2009年逐8天数据&#xff0c;值域是-100-689之间&#xff0c;数据类型为32bit整型。该产品经过遥感数据获取、计算归一化植被指数、解译植被类…

测量模拟量的优选模块:新型设备M-SENS3 8

| 具有8路自由选择通道的新型设备M-SENS3 8 IPETRONIK推出的模拟量测量设备——M-SENS3 8是新一代设备的新成员。该模块具有8个通道&#xff0c;能够自由选择测量模式&#xff0c;不仅支持高精度电压和电流的测量&#xff0c;还新增了频率测量模式。各通道分辨率高达18位&…

Selenium常用命令(python版)

日升时奋斗&#xff0c;日落时自省 目录 1、Selenium 2、常见问题 1、Selenium 安装Python和配置环境没有涉及 注&#xff1a;如有侵权&#xff0c;立即删除 首先安装selenium包&#xff0c;安装方式很简单 pip install selenium 注:我这里已经安装好了&#xff0c;所以…

spring boot集成Knife4j

文章目录 一、Knife4j是什么&#xff1f;二、使用步骤1.引入依赖2.新增相关的配置类3.添加配置信息4.新建测试类5. 启动项目 三、其他版本集成时常见异常1. Failed to start bean ‘documentationPluginsBootstrapper2.访问地址后报404 一、Knife4j是什么&#xff1f; 前言&…

弘君资本股市行情:股指预计保持震荡上扬格局 关注汽车、银行等板块

弘君资本指出&#xff0c;近期商场体现全体分化&#xff0c;指数层面上看&#xff0c;沪指一路震动上行&#xff0c;创出年内新高&#xff0c;创业板指和科创50指数体现相对较弱&#xff0c;依然是底部震动走势。从盘面体现上看&#xff0c;轮动依然是当时商场的主基调&#xf…

逻辑分析仪 - 采样率/采样深度

采样深度&#xff08;Sampling Depth&#xff09; 采样深度指的是逻辑分析仪在一次捕获过程中可以记录的最大样本数量。简单来说&#xff0c;采样深度越大&#xff0c;逻辑分析仪可以记录的数据量就越多。这对于分析长时间的信号变化或复杂的信号序列非常重要。 采样率&#…

WEB攻防【2】——ASPX/.NET项目/DLL反编译/未授权访问/配置调试报错

ASP&#xff1a;windowsiisaspaccess .net&#xff1a;windowsiisaspxsqlserver IIS上的安全问题也会影响到 WEB漏洞&#xff1a;本身源码上的问题 服务漏洞&#xff1a;1、中间件 2、数据库 3、第三方软件 #知识点: 1、.NET:配置调试-信息泄绵 2、.NET:源码反编译-DLL…

使用Flask ORM进行数据库操作的技术指南

文章目录 安装Flask SQLAlchemy配置数据库连接创建模型类数据库操作插入数据查询数据更新数据删除数据 总结 Flask是一个轻量级的Python Web框架&#xff0c;其灵活性和易用性使其成为开发人员喜爱的选择。而ORM&#xff08;对象关系映射&#xff09;则是一种将数据库中的表与面…