前綴和應用: 總和=k的子陣列有幾個 Subarray Sum Equals K_Leetcode #560

更新於 發佈於 閱讀時間約 6 分鐘

題目敘述

題目會給我們一個輸入陣列nums,和一個指定的k值。

請問,在輸入陣列nums中,有幾個子陣列的元素總合恰好為k ?

例如: nums = [1,2,3], k = 3

則有兩個子陣列的元素總合為3,分別是[1,2] 和 [3]


如果是第一次聽到或接觸前綴和prefix的同學,請先參考這篇解題教學文章,因為接下來,我們會用到前綴和的觀念去解題。

題目的原文敘述


測試範例

Example 1:

Input: nums = [1,1,1], k = 2
Output: 2

有兩個子陣列的元素總合為 k = 2
[1,1][1,1]
前兩個1 和 後兩個1

Example 2:

Input: nums = [1,2,3], k = 3
Output: 2

有兩個子陣列的元素總合為 k​ = 3
[1,2][3]

約束條件

Constraints:

  • 1 <= nums.length <= 2 * 10^4

輸入陣列nums的長度介於1~兩萬之間。

  • -1000 <= nums[i] <= 1000

每個陣列元素值界於 -1000 ~ 1000

  • -10^7 <= k <= 10^7

指定的k值介於 負一千萬~正一千萬 之間。


演算法 從第一直覺的想法,演化到進階的前綴和高速計算區間和(以前正好學過)

第一直覺的想法可能是直接用兩層迴圈去找,第一層找所有可能的左端點left,第二層找所以可能的右端點right,再計算[left, right]區間內的元素總合是否剛好=k,去累積子陣列個數。

這個想法算是暴力展開法,面對題目的陣列輸入長度高達 兩萬 個完素,暴力搜索將付出平方等級的代價,高達 10*8 等級的搜索成本,明顯會有太慢time-out超過平台限制的風險。


這邊,我們會借用前面學過透過prefix sum 來高速計算 range sum的技巧,來降低演算法的複雜度,題生程式執行效率。


一開始,先建立一個字典,鍵值是prefix sum,value則是這個prefix sum的出現次數。

接著,建造一個迴圈,從左到右,逐漸更新前綴和 prefix sum,接著檢查
prefix sum - k 這個比較小的值以前有沒有出現過?
假如有,代表已經找到子陣列=k的區間,累加子陣列=k的的出現次數

接著更新當下的prefix sum的出現次數,寫到字典裡面。

依照同樣的模式,反覆迭代。

最後,子陣列=k的的出現次數計數器的值,就是題目所求,也是最終回傳的答案


以下圖說明演算法精神,假設今天題目給的k值=6

當下我正好計算到箭頭指向的地方,目前累積的prefix sum前綴和 = S + 6

那我們就去檢查 (S + 6 - k) = S + 6 - 6 = S 這個值以前有沒有出現過?

如果有,S以前出現在淺藍色的那個格子,代表從上一次出現的地方到目前這個位置,剛好構成一組子陣列,區間內的元素總合 = k = 3 + 2 + 1 = 6

image

image



程式碼 用 前綴和 高速計算 區間和(以前正好學過)

class Solution:
def subarraySum(self, nums: List[int], k: int) -> int:

# prefix sum
cur_prefix_sum = 0

# counter of subarray with sum = k
counter = 0

# key: subarray prefix sum
# value: occurrence of given prefix sum
prefixSum_map = defaultdict(int)

# prefix summation 0 is reached by selecting nothing
prefixSum_map[0] = 1

for number in nums:

# update prefix sum from first element to current position
cur_prefix_sum += number

# if curPrefixSum - k exist in map, then sub array with sum = k must exist in somewhere
if (cur_prefix_sum - k) in prefixSum_map:
counter += prefixSum_map[cur_prefix_sum - k]

# update occurrence of curPrefixSum
prefixSum_map[cur_prefix_sum] += 1


return counter

複雜度分析 用 前綴和 高速計算 區間和(以前正好學過)

令n=陣列總長度

時間複雜度:

從左到右線性掃描,逐步更新prefix sum,並且寫入到字典裡面,並且動態更新子陣列=k的出現次數,所需時間為O(n)。

空間複雜度:

成本耗費在字典prefixSum_map,最多儲存n種不同的前綴和,所耗空間最大為O(n)。


關鍵知識點

建造一個迴圈,從左到右,逐漸更新前綴和 prefix sum,接著檢查
prefix sum - k 這個比較小的值以前有沒有出現過?
假如有,代表已經找到子陣列=k的區間,累加子陣列=k的的出現次數

也可以順便複習 Range Sum這題,加強鞏固關於使用 前綴和 高速計算區間和的技巧


Reference:

[1] Subarray sum Equals k, O(n) by prefix sum + hashMap

avatar-img
90會員
425內容數
由有業界實戰經驗的演算法工程師, 手把手教你建立解題的框架, 一步步寫出高效、清晰易懂的解題答案。 著重在讓讀者啟發思考、理解演算法,熟悉常見的演算法模板。 深入淺出地介紹題目背後所使用的演算法意義,融會貫通演算法與資料結構的應用。 在幾個經典的題目融入一道題目的多種解法,或者同一招解不同的題目,擴展廣度,並加深印象。
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
題目敘述 題目會給定我們一個字串s,和一組字庫wordDict。 問我們能不能透過字串串接的方式,從字庫裡面的字拼成原本的字串s? 可以的話,返回True。 無解的話,返回False。 註: 題目還允許重複使用字庫裡面的字去串接。
題目敘述 題目會給定一個指定高度和寬的方格版,還有一顆小球的起始位置,和最大移動步數。 小球每一步可以選擇向上、下、左、右移動一格,請問小球能走到方格版界外的路徑方法數總共有幾種? 方法數可能很大,題目要求,最後回傳答案時,先對10^9+7做除法取餘數再回傳。 題目的原文敘述 約束條件
這題也是滿經典的DP動態規劃教學案例和題目,就順便複習一下吧。 題目敘述 題目會給我們兩個字串text1, text2。 要求我們找出兩個字串的最長共同子序列,並且返回最長共同子序列的長度。 如果彼此沒有共同子序列,則返回0。 題目的原文敘述 測試範例 Example 1: In
這題也算是Leetcode 上經典的DP考題之一,也是很好的DP邏輯思考練習題。 題目敘述 題目會給我們一個nums陣列,分別代表每棟房屋的價值,也就是房屋內有的現金數量。 題目敘述給的情境是假想盜賊要偷東西,限制是相鄰的兩棟房屋不能一起偷,只能選擇其中一棟,否則就會觸發警報器。 請問怎麼選
題目敘述 題目會給我們一個二維陣列matrix,分別代表每個格子的成本,請問我們從最頂端到底部的下墜路徑的最小成本總和是多少? 每次下墜到下一排的時候,可以有三種選擇: 1.往左下角移動。 2.往正下方移動。 3.往右下角移動。 題目的原文敘述 測試範例 Example 1:
題目敘述 題目會告訴我們一組英文和數字之間的轉換編碼規則,還有一個輸入字串s,問我總共有多少合法的解碼方式? 要特別留意,輸入字串可能包含有leading zero,導致無法解碼。 轉換規則如下: A <-> 1 B <-> 2 C <-> 3 ... Z <-> 26 詳細的題
題目敘述 題目會給定我們一個字串s,和一組字庫wordDict。 問我們能不能透過字串串接的方式,從字庫裡面的字拼成原本的字串s? 可以的話,返回True。 無解的話,返回False。 註: 題目還允許重複使用字庫裡面的字去串接。
題目敘述 題目會給定一個指定高度和寬的方格版,還有一顆小球的起始位置,和最大移動步數。 小球每一步可以選擇向上、下、左、右移動一格,請問小球能走到方格版界外的路徑方法數總共有幾種? 方法數可能很大,題目要求,最後回傳答案時,先對10^9+7做除法取餘數再回傳。 題目的原文敘述 約束條件
這題也是滿經典的DP動態規劃教學案例和題目,就順便複習一下吧。 題目敘述 題目會給我們兩個字串text1, text2。 要求我們找出兩個字串的最長共同子序列,並且返回最長共同子序列的長度。 如果彼此沒有共同子序列,則返回0。 題目的原文敘述 測試範例 Example 1: In
這題也算是Leetcode 上經典的DP考題之一,也是很好的DP邏輯思考練習題。 題目敘述 題目會給我們一個nums陣列,分別代表每棟房屋的價值,也就是房屋內有的現金數量。 題目敘述給的情境是假想盜賊要偷東西,限制是相鄰的兩棟房屋不能一起偷,只能選擇其中一棟,否則就會觸發警報器。 請問怎麼選
題目敘述 題目會給我們一個二維陣列matrix,分別代表每個格子的成本,請問我們從最頂端到底部的下墜路徑的最小成本總和是多少? 每次下墜到下一排的時候,可以有三種選擇: 1.往左下角移動。 2.往正下方移動。 3.往右下角移動。 題目的原文敘述 測試範例 Example 1:
題目敘述 題目會告訴我們一組英文和數字之間的轉換編碼規則,還有一個輸入字串s,問我總共有多少合法的解碼方式? 要特別留意,輸入字串可能包含有leading zero,導致無法解碼。 轉換規則如下: A <-> 1 B <-> 2 C <-> 3 ... Z <-> 26 詳細的題
你可能也想看
Google News 追蹤
Thumbnail
隨著理財資訊的普及,越來越多台灣人不再將資產侷限於台股,而是將視野拓展到國際市場。特別是美國市場,其豐富的理財選擇,讓不少人開始思考將資金配置於海外市場的可能性。 然而,要參與美國市場並不只是盲目跟隨標的這麼簡單,而是需要策略和方式,尤其對新手而言,除了選股以外還會遇到語言、開戶流程、Ap
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
這篇內容,將會講解什麼是陣列,以及與陣列相關的知識。包括陣列的簡介、陣列的資料限制、陣列的維度、一維陣列、二維陣列。
Thumbnail
計數原理的題目們,以及上次排組的解析。
Thumbnail
這篇文章介紹了排列和組閤中的錯位排列和排容原理,並提供了一種相對樸實的解題方法。透過例子詳細解釋了選擇情況下的數學原理,讓讀者能夠理解並吸收。文章通過課堂上難以推敲的題目,提出了一個相對簡單的方式來解題。 圖片選自@pngtree
Thumbnail
分享在網路上看到的陣列題目。通常 for...of 的 value 是陣列中的每個值,那如果我們在迭代中對陣列操作會發生什麼事? 題目來源:https://x.com/_jayphelps/status/1774640511158022335?s=20
Thumbnail
最近每天都有同學在解題社群提問這類型的問題,有些同學甚至po出解答來提問,表示看了解答卻還是看不懂,畢竟有時候「詳解」也沒辦法完整表達所有觀念。 排列組合是一門龐大的章節,許多人聞排組而色變,但排列組合的本質其實還是「窮舉法」,也就是把全部的可能通通列出來,只是很多地方我們可以透過計算讓窮舉變得更
Thumbnail
隨著理財資訊的普及,越來越多台灣人不再將資產侷限於台股,而是將視野拓展到國際市場。特別是美國市場,其豐富的理財選擇,讓不少人開始思考將資金配置於海外市場的可能性。 然而,要參與美國市場並不只是盲目跟隨標的這麼簡單,而是需要策略和方式,尤其對新手而言,除了選股以外還會遇到語言、開戶流程、Ap
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
這篇內容,將會講解什麼是陣列,以及與陣列相關的知識。包括陣列的簡介、陣列的資料限制、陣列的維度、一維陣列、二維陣列。
Thumbnail
計數原理的題目們,以及上次排組的解析。
Thumbnail
這篇文章介紹了排列和組閤中的錯位排列和排容原理,並提供了一種相對樸實的解題方法。透過例子詳細解釋了選擇情況下的數學原理,讓讀者能夠理解並吸收。文章通過課堂上難以推敲的題目,提出了一個相對簡單的方式來解題。 圖片選自@pngtree
Thumbnail
分享在網路上看到的陣列題目。通常 for...of 的 value 是陣列中的每個值,那如果我們在迭代中對陣列操作會發生什麼事? 題目來源:https://x.com/_jayphelps/status/1774640511158022335?s=20
Thumbnail
最近每天都有同學在解題社群提問這類型的問題,有些同學甚至po出解答來提問,表示看了解答卻還是看不懂,畢竟有時候「詳解」也沒辦法完整表達所有觀念。 排列組合是一門龐大的章節,許多人聞排組而色變,但排列組合的本質其實還是「窮舉法」,也就是把全部的可能通通列出來,只是很多地方我們可以透過計算讓窮舉變得更