[Lc滑動窗口_1] 長度最小的數組 | 無重復字符的最長子串 | 最大連續1的個數 III | 將 x 減到 0 的最小操作數

目錄

1. 長度最小的字數組

題解

代碼

?2.無重復字符的最長子串

題解

代碼

3.最大連續1的個數 III

題解

代碼

4.將 x 減到 0 的最小操作數

題解

代碼


1. 長度最小的字數組

題目鏈接:209.長度最小的字數組

題目分析:

給定一個含有 n 個 正整數 的數組和一個正整數 target

找出該數組中滿足其總和大于等于 target 的長度最小的 子數組 [numsl, numsl+1, ..., numsr-1, numsr] ,并返回其長度如果不存在符合條件的子數組,返回 0

子數組:是連續的!!!)

示例 1:

輸入:target = 7, nums = [2,3,1,2,4,3]
輸出:2
解釋:子數組 [4,3] 是該條件下的長度最小的子數組。

示例 2:

輸入:target = 4, nums = [1,4,4]
輸出:1

題解

  • 注意題目說的是 正整數 數組,說明數組里面的數是大于等于0的數。
  • 因此這道題我們有一種優化的方法。
  • 題目讓找連續的子數組和>=target,并且長度最小。有很多種情況,但是我們選擇的是 最小長度。

算法原理:

不管什么題,首先我們一定會先想到的是 暴力求解,因為只有暴力求解出來了,我們就可以在暴力求解的基礎上進行優化~

解法一:暴力枚舉出所有的子數組的和

兩層for循環,O(N^2)

for(int i=0;i<n;i++)for(int j=i;j<n;j++)sum+=num[j];
//固定一個數,挨個往后加if(sum>=target)min(len,j-i+1)

注意到此時暴力枚舉是有優化的。

題目說的是一個 正整數數組,都是大于等于0的數,這個 sum是呈現出遞增的狀態的,單調遞增!

在暴力求解中,此時right還要++,但是注意題目本來要求的就是 最小長度

此時sum在加上往上走了一步的right的num[right],一定是滿足sum>=target,但是len變成5了,一定不會是最終結果

因此當條件已經滿足sum>=target ,right就不用動了。right后面也就不用再枚舉了。

那現在讓 left+1,right和left指向同一下標,然后再重復上面過程,那有個問題,這段區間的和能不能直接算出來?

  • 當然可以。
  • 現在sum=8,我只需要把讓sum減去num[left],不就是現在left和right所在的區間和算出來嗎。
  • 沒有必要讓right傻傻的回退然后重新加。因此right不動,更新sum=6.

因此我們從暴力枚舉中發現兩個優化:

  • 一個是right 滿足后,后面不用枚舉
  • 一個left++,right不用回退,

所以我們可以利用單調性,使用雙指針優化。


解法二:利用單調性,使用 “同向雙指針來優化

當我們在暴力枚舉的策略中發現left和right都是從左向右一個方向移動,我們就稱為這兩個指針叫做同向指針。同向雙指針又稱為滑動窗口。

什么是滑動窗口?

本質上是 “同向雙指針”,left從左到右移動,right不回退,從左到右移動,用left和right一直 維護這個區間的和,然后這兩個指針從左向右移動的過程非常像一個窗口在這個數組里滑來滑去。

什么時候用滑動窗口?

利用單調性,用滑動窗口解決問題。

當我們發現在暴力求解時,兩個指針都可以做到 不回退,都是向同一個方向移動的時候,此時就可以用滑動窗口。

滑動窗口怎么用?

  1. 初始left=0,right=0,充當窗口左端點,右端點。用left,right標記窗口左區間,右區間。
  2. 右窗口(++right)(右值進窗口)
  3. 判斷
    • 根據判斷決定是否 左窗口(++left)(左值出窗口)
  1. 更新結果
    • 2,3都有可能會更新結果,看題目要求

左窗口,判斷,右窗口一直循環,直到right 超過區間長度結束更新結果看題目要求(右窗口,左窗口都有可能),。

滑動窗口正確性

  • 暴力枚舉肯定對的,因為已經把所有子數組的情況都找出來了。
  • 雖然滑動窗口并沒有把沒有把所有情況都枚舉出來,但是這里利用單調性,規避了沒有必要的枚舉
  • 雖然沒有把所有情況真正枚舉出來,但是已經判斷出有些子數組不是最終結果,已經把所有結果都考慮進來了,所以這種策略是跟暴力枚舉是一樣正確的。

滑動窗口時間復雜度

進窗口是一個循環,判斷也是一個循環。

兩層循環套在一起。可能會覺得時間復雜度O(N^2),但是不能看代碼算時間復雜度,要看實際情況分析實際復雜度。

實際我們只會讓right向前移動,left也向前移動,即使時最壞情況,right移動到最后一個元素,left 也移動到最后一個元素,因為單調性,總共也就操作了 n+n=2n 次 整體時間復雜度O(N)


代碼

要考慮到,棧溢出(heap-buffer-overflow) 的邊界情況

可詳見前文

在Leetcode中,無窮大和無窮小分別怎么表示

C/C++中可以使用INT_MAX和INT_MIN


?2.無重復字符的最長子串

題目鏈接:3. 無重復字符的最長子串

題目分析:

給定一個字符串 s ,請你找出其中不含有重復字符的 最長 子串 的長度。

示例 1:

輸入: s = "abcabcbb"
輸出: 3 
解釋: 因為無重復字符的最長子串是 "abc",所以其長度為 3。

示例 2:

輸入: s = "bbbbb"
輸出: 1
解釋: 因為無重復字符的最長子串是 "b",所以其長度為 1。

示例 3:

輸入: s = "pwwkew"
輸出: 3
解釋: 因為無重復字符的最長子串是 "wke",所以其長度為 3。請注意,你的答案必須是 子串 的長度,"pwke" 是一個子序列,不是子串。

子串和子數組都是連續的

??? 模擬分析示例:

題解

算法原理:

首先還是暴力枚舉,然后根據暴力枚舉進行優化。

  • 以下面為例,兩層for循環,但是下面找到的結果都是我們站在上帝角度,編譯器并不知到什么時候結束。
  • 一般對應判斷是否有重復元素,我們都可以用哈希表來解決問題。
  • 使用哈希表,判斷是否有重復元素,比如讓你判斷一個數組是否有重復,或者兩個數組是否有重復都可以用哈希映射!

解法一:暴力枚舉+哈希表(判斷字符是否重復出現)

O(N^2)

根據解法一做優化,定義一個left,right指針。當right走到有重復的元素后,已經找到一個字串,其中left到right區間每個元素都已經進入hash表。

此時left向前走一步,但是這個區間還是有重復元素,因此left要走到沒有重復的區間才行,

然后這個時候以前做法是right回退然后重新往下走,但是這里left到right區間元素本來就在hash表里

因此就不需要right回退了,而是向right繼續向前走。然后重復上面過程,直到right走到結尾。結束~

這不就是滑動窗口的思想嗎。雙向指針,left往前走,right不回退一直往前走

解法二:利用規律,使用 “滑動窗口” 解決問題

  1. left=0,right=0
  2. 進窗口
  3. 判斷
    • 出窗口
  1. 更新結果

進窗口、判斷、出窗口,更新結果是一個大循環過程。直到right到結尾循環結束。

其中判斷、出窗口是一個小循環(直到跳出重復字符)。不過時間復雜度還是O(N).

注意:

  • 更新結果可能在? 進窗口后,判斷后,出窗口后,判斷后任意一個地方,看題目要求

本題:

  • 進窗口 ->-> 讓字符進入哈希表
  • 判斷-> 窗口內出現重復元素
  • 出窗口-> 從哈希表中刪除該字符

代碼

class Solution {
public:int lengthOfLongestSubstring(string s) {//!!if(s.empty()) return 0; // 處理空輸入vector<char> str;for(char c:s) str.push_back(c);int left=0,right=0,n=str.size(),len=0;//unordered_set ret;unordered_set<char> ret;while(right<n){
//先檢查while(ret.count(str[right])){ret.erase(str[left]);left++;//利用了連續性//表中 發現了右元素已存在//要在左邊 進行跳過}
//不存在 就插入ret.insert(str[right]);len=max(len,right-left+1);right++;}return len;}
};

總結一下:

利用單調性,使用 雙指針 解決問題。

  • 一般left和right,一個指向數組最左邊,一個指向數組最右邊,然后一次可以排除一批,再然后left++,–right,兩個指針是對撞的。

這里利用單調性或者利用規律,使用 滑動窗口 解決問題

  • 滑動窗口也使用雙指針解決問題,不過left一直從前往后走,right不回退從前往后走,兩個指針是同向的。因此滑動窗口本質其實是 同向雙指針。

3.最大連續1的個數 III

題目鏈接:1004. 最大連續1的個數 III

題目分析:

給定一個二進制數組 nums 和一個整數 k,假設最多可以翻轉 k0 ,則返回執行操作后 數組中連續 1 的最大個數

示例 1:

輸入:nums = [1,1,1,0,0,0,1,1,1,1,0], K = 2
輸出:6
解釋:[1,1,1,0,0,1,1,1,1,1,1]
粗體數字從 0 翻轉到 1,最長的子數組長度為 6。

示例 2:

輸入:nums = [0,0,1,1,0,0,1,1,1,0,1,1,0,0,0,1,1,1,1], K = 3
輸出:10
解釋:[0,0,1,1,1,1,1,1,1,1,1,1,0,0,0,1,1,1,1]
粗體數字從 0 翻轉到 1,最長的子數組長度為 10。

題解

題目說的翻轉實際上是把0變成1的意思,最多翻轉K次,說明小于等于K都是可以的。

拿到題我們開始肯定想的是暴力求解。

如果直接暴力求解,遇到0->1了,那下一次在遍歷就有問題了。

因此我們換一個思路。這道題不是讓轉化后最大連續1的個數嗎。

我們轉化為:找出最長的子數組,數組里0的個數不超過K個,這個數組里面0一定能夠轉化成1。

算法原理:

解法一:暴力枚舉+zero計數器

偽代碼,兩層for循環,統計zero的個數,滿足zero>k,統計此時數組長度,然后重新進入循環,注意每次zero都清0

for(int i=0;i<n;++i)for(int j=i;j<n;++j)//雙指針 查找出一段區間if(nums[j]==0)++zero;if(zero>k)ret=max(ret,right-left+1)

然后我們根據暴力枚舉,看看有沒有優化的可能。

定義兩個指針left,right,right走到zero>k的位置,zero=3,大于k。

按照暴力求解left++,然后right回溯然后重新往后走。但是我們發現沒有必要,現在left往前走一步,你會發現,right還是停留在老位置,這個區間不用在管的,直接丟棄。

因此,讓left一直走到zero<=k的位置。然后 right也根本不用回溯 然后在重新走,而是直接往后走就行了。

根據上面的發現,當在暴力枚舉中,發現left,right是同向移動的,利用這個規律,使用滑動窗口解決問題

解法二:利用規律,使用滑動窗口

  1. left=0,right=0
  2. 進窗口
  3. 判斷
    • 出窗口
  1. 更新結果

進窗口 -> 如果是1,不理會。如果是0,計數器+1

判斷 -> zero>k

出窗口 -> 如果是1,不理會。如果是0,計數器-1

更新結果:在判斷之后在更新

代碼

class Solution {
public:int longestOnes(vector<int>& nums, int k) {int left = 0, right = 0, zero = 0;int n = nums.size(), len = 0;while (right < n) {//進 窗口if (nums[right] == 0)zero++;while (zero > k) {//循環判斷if (nums[left] == 0)zero--;left++;}len = max(len, right - left + 1);right++;}return len;}
};

4.將 x 減到 0 的最小操作數

題目鏈接:1658. 將 x 減到 0 的最小操作數

題目分析:

給你一個整數數組 nums 和一個整數 x 。每一次操作時,你應當移除數組 nums 最左邊或最右邊的元素,然后從 x 中減去該元素的值。請注意,需要 修改 數組以供接下來的操作使用。

如果可以將 x 恰好 減到 0 ,返回 最小操作數 ;否則,返回 -1

示例 1:

輸入:nums = [1,1,4,2,3], x = 5
輸出:2
解釋:最佳解決方案是移除后兩個元素,將 x 減到 0 。

示例 2:

輸入:nums = [5,6,7,8,9], x = 4
輸出:-1

示例 3:

輸入:nums = [3,2,20,1,1,3], x = 10
輸出:5
解釋:最佳解決方案是移除后三個元素和前兩個元素(總共 5 次操作),將 x 減到 0 。

題解

這道題讓每次從數組左右兩邊移除一個數,然后就是一個新的數組,然后再從新的數組再從左右兩邊移除一個數。

  • 但是如果真的硬著頭皮開始做,其實是很困難的。
  • 并不知道每次是從最左邊走還是最右邊找。有可能這次左邊下次右邊或者還是左邊,情況太復雜了。

因此我們可以利用 正難則反 的思想

  • 正對面解題太難,那就想對立面,換個思路。
  • 不是每次從左右兩端找一個數嗎,那可能找到情況就是a+b=x,a、b什么情況都要,但是中間這個連續區間的和不也是確定的嗎sum-x
  • 也就是這道題我們轉換成,找出最長的子數組長度,所有元素的和正好等于sum-x,然后數組總長減去這段子區間長度不就是問題答案嗎
  • 如果沒找到說明這個數組不存在將x減到0的數,直接返回-1

解法一:暴力求解

初始left,right指向同一下標,當right走到和大于target的時候,left往前走

按照暴力求解,right要回到和left相同下標,然后right在重新往前走,直到再次走到和大于target的地方停下來,然后重復上面過程。

  • 但是今天這里不需要right回溯,因為right回溯后重新走到下面的位置,因為left已經往前走了,這段區間的和肯定是更小了
  • 因此就不需要right回溯了。要么right不動,要么right往后走。
  • 同向雙指針 ----> 本質就是滑動窗口

解法二:使用滑動窗口

代碼


class Solution {
public:
int minOperations(vector<int>& nums, int x)
{if(nums.empty()) return -1;int sum=0;for(auto c:nums)sum+=c;// 新增邊界條件處理if (sum == x) return nums.size();  // 整個數組和正好等于xif (sum < x) return -1;            // 總和不足,無法達成目標int target=sum-x,len=0;int left=0,right=0,n=nums.size(),add=0;while(right<n){add+=nums[right];while(add>target){add-=nums[left];left++;}if(add==target)len=max(len,right-left+1);right++;}//!!!len>0      return (len > 0) ? (nums.size() - len) : -1;
}
};

測試樣例跑不全時,要注意對 邊界情況 的處理

  • 若不存在這樣的子數組(len = 0

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/71172.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/71172.shtml
英文地址,請注明出處:http://en.pswp.cn/web/71172.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

數據集筆記:新加坡 地鐵(MRT)和輕軌(LRT)票價

數據連接 data.gov.sg 2024 年 12 月 28 日起生效的新加坡地鐵票價 該數據集包含 MRT 和 LRT 票價的信息&#xff0c;包括&#xff1a; 票價類型&#xff08;Fare Type&#xff09;&#xff1a;成人票、學生票、老年人票、殘障人士票等。適用時間&#xff08;Applicable Tim…

湘潭大學計算機復試詳細攻略(調劑)

一&#xff0c;寫在前面的話 ① 首先&#xff0c;能完成考試初試來到這里的都是勇士。不管結果如何&#xff0c;不管成績如何。我都在這里真心的祝福你以后一帆風順。 ② 目前學歷貶值嚴重&#xff0c;如果是成績不理想的話&#xff0c;我建議能工作就去工作&#xff0c;工作不…

【前端基礎】Day 3 CSS-2

目錄 1. Emmet語法 1.1 快速生成HTML結構語法 1.2 快速生成CSS樣式語法 2. CSS的復合選擇器 2.1 后代選擇器 2.2 子選擇器 2.3 并集選擇器 2.4 偽類選擇器 2.4.1 鏈接偽類選擇器 2.4.2 focus偽類選擇器 2.5 復合選擇器總結 3. CSS的元素顯示模式 3.1 什么是元素顯示…

不同數據類型在數據庫和編程語言之間的對應關系表

不同數據類型在數據庫和編程語言之間的對應關系表 MySql 與 C# MySqlC#varcharstringbigintlongbigint unsignedulongintintint unsigneduintsmallintshortsmallint unsignedushortVARCHAR(36)GuidsmalldatetimeDateTimedateDateTimedatetimeDateTimetimestampDateTimefloatf…

RabbitMQ操作實戰

1.RabbitMQ安裝 RabbitMQ Windows 安裝、配置、使用 - 小白教程-騰訊云開發者社區-騰訊云下載erlang&#xff1a;http://www.erlang.org/downloads/https://cloud.tencent.com/developer/article/2192340 Windows 10安裝RabbitMQ及延時消息插件rabbitmq_delayed_message_exch…

DeepSeek教unity------UI元素長按響應

主要功能說明&#xff1a; ?長按檢測&#xff1a;通過記錄指針按下的時間&#xff0c;判斷是否達到 longClickTime&#xff0c;從而觸發長按事件。?狀態管理&#xff1a;使用 StateEnum 枚舉管理點擊項的當前狀態&#xff08;未按下、按下等待長按、長按已觸發&#xff09;。…

【北京迅為】itop-3568 開發板openharmony鴻蒙燒寫及測試-第2章OpenHarmony v3.2-Beta4版本測試

瑞芯微RK3568芯片是一款定位中高端的通用型SOC&#xff0c;采用22nm制程工藝&#xff0c;搭載一顆四核Cortex-A55處理器和Mali G52 2EE 圖形處理器。RK3568 支持4K 解碼和 1080P 編碼&#xff0c;支持SATA/PCIE/USB3.0 外圍接口。RK3568內置獨立NPU&#xff0c;可用于輕量級人工…

stm32hal庫尋跡+藍牙智能車(STM32F103C8T6)

簡介: 這個小車的芯片是STM32F103C8T6&#xff0c;其他的芯片也可以照貓畫虎,基本配置差不多,要注意的就是,管腳復用,管腳的特殊功能,(這點不用擔心,hal庫每個管腳的功能都會給你羅列,很方便的.)由于我做的比較簡單,只是用到了幾個簡單外設.主要是由帶霍爾編碼器電機的車模,電機…

SQL命令詳解之操作數據庫

操作數據庫 SQL是用于管理和操作關系型數據庫的標準語言。數據庫操作是SQL的核心功能之一&#xff0c;主要用于創建、修改和刪除數據庫對象&#xff0c;如數據庫、表、視圖和索引等。以下是SQL中常見的數據庫操作命令及其功能簡介&#xff1a; 1. 查詢數據庫 查詢所有的數據庫…

Go紅隊開發—編解碼工具

文章目錄 開啟一個項目編解碼工具開發Dongle包Base64編解碼摩斯密碼URL加解密AES加解密 MD5碰撞工具開發 開啟一個項目 這作為補充內容&#xff0c;可忽略直接看下面的編解碼&#xff1a; 一開始用就按照下面的步驟即可 1.創建一個文件夾&#xff0c;你自己定義名字(建議只用…

Starrocks入門(二)

1、背景&#xff1a;考慮到Starrocks入門這篇文章&#xff0c;安裝的是3.0.1版本的SR&#xff0c;參考&#xff1a;Starrocks入門-CSDN博客 但是官網的文檔&#xff0c;沒有對應3.0.x版本的資料&#xff0c;卻有3.2或者3.3或者3.4或者3.1或者2.5版本的資料&#xff0c;不要用較…

工程化與框架系列(10)--微前端架構

微前端架構 &#x1f3d7;? 微前端是一種將前端應用分解成更小、更易管理的獨立部分的架構模式。本文將詳細介紹微前端的核心概念、實現方案和最佳實踐。 微前端概述 &#x1f31f; &#x1f4a1; 小知識&#xff1a;微前端的核心理念是將前端應用分解成一系列獨立部署、松耦…

SwiftUI之狀態管理全解析

文章目錄 引言一、`@State`1.1 基本概念1.2 初始化與默認值1.3 注意事項二、`@Binding`2.1 基本概念2.2 初始化與使用2.3 注意事項三、`@ObservedObject`3.1 基本概念3.2 初始化與使用3.3 注意事項四、`@EnvironmentObject`4.1 基本概念4.2 初始化與使用4.3 注意事項五、`@Stat…

Redis 高可用性:如何讓你的緩存一直在線,穩定運行?

&#x1f3af; 引言&#xff1a;Redis的高可用性為啥這么重要&#xff1f; 在現代高可用系統中&#xff0c;Redis 是一款不可或缺的分布式緩存與數據庫系統。無論是提升訪問速度&#xff0c;還是實現數據的高效持久化&#xff0c;Redis 都能輕松搞定。可是&#xff0c;當你把 …

面試題:說一下你對DDD的了解?

面試題:說一下你對DDD的了解? 在面試中,關于 DDD(領域驅動設計,Domain-Driven Design) 的問題是一個常見的技術考察點。DDD 是一種軟件設計方法論,旨在通過深入理解業務領域來構建復雜的軟件系統。以下是一個清晰、詳細的回答模板,幫助你在面試中脫穎而出: DDD 的定義…

Redis---緩存穿透,雪崩,擊穿

文章目錄 緩存穿透什么是緩存穿透&#xff1f;緩存穿透情況的處理流程是怎樣的&#xff1f;緩存穿透的解決辦法緩存無效 key布隆過濾器 緩存雪崩什么是緩存雪崩&#xff1f;緩存雪崩的解決辦法 緩存擊穿什么是緩存擊穿&#xff1f;緩存擊穿的解決辦法 區別對比 在如今的開發中&…

Android Logcat 高效調試指南

工具概覽 Logcat 是 Android SDK 提供的命令行日志工具&#xff0c;支持靈活過濾、格式定制和實時監控&#xff0c;官方文檔詳見 Android Developer。 基礎用法 命令格式 [adb] logcat [<option>] ... [<filter-spec>] ... 執行方式 直接調用&#xff08;通過ADB守…

【定昌Linux系統】部署了java程序,設置開啟啟動

將代碼上傳到相應的目錄&#xff0c;并且配置了一個.sh的啟動腳本文件 文件內容&#xff1a; #!/bin/bash# 指定JAR文件的路徑&#xff08;如果JAR文件在當前目錄&#xff0c;可以直接使用文件名&#xff09; JAR_FILE"/usr/local/java/xs_luruan_client/lib/xs_luruan_…

Java 8 中,可以使用 Stream API 和 Comparator 對 List 按照元素對象的時間字段進行倒序排序

文章目錄 引言I 示例對象II List 按時間字段倒序排序: 使用 `Stream` 和 `Comparator` 排序方法 1:使用 `Comparator.comparing`方法 2:使用 `Comparator.reversed`方法 3:自定義 `Comparator`輸出結果III 注意事項**時間字段類型**:**空值處理**:IV 總結引言 案例:在線用…

jvm內存模型,類加載機制,GC算法,垃圾回收器,jvm線上調優等常見的面試題及答案

JVM內存模型 JVM內存模型包括哪些區域 答案&#xff1a;JVM內存模型主要包括以下區域&#xff1a; 程序計數器&#xff1a;是一塊較小的內存空間&#xff0c;它可以看作是當前線程所執行的字節碼的行號指示器&#xff0c;用于記錄正在執行的虛擬機字節碼指令的地址。Java虛擬機…