[Leetcode Pattern] Sliding Window

Pattern 精髓

所謂 window 就是指一個範圍,例如字串 or Array 裡的區間,而 Sliding 就是只可以動態調整 window 大小,Sliding window 最主要是避免重複計算

幾個構成的條件

  • 透過操作 2 個 point 構成 window ,計算來達成目的
  • window 的搜索範圍是一串連續的資料

例題帶入講解

最經典的就是 leetcode 的 209. Minimum Size Subarray Sum

題目是要我們找到一個 Array 例如 [2,3,1,2,4,3] 裡面最少的 subarray 且 subarray 的 sum要大於等於 target ,假設 target = 7 即 subarray 為 [4, 3] 時滿足條件,所以 return subarray 的 count 2

首先看一下暴力解:

class Solution {
    func minSubArrayLen(_ target: Int, _ nums: [Int]) -> Int {
        var ans = Int.max
        for i in 0..<nums.count {
            print("<i start:\(i)>")
            for j in i..<nums.count {
                var sum = 0
                for k in i...j {
                    print("sum += \(nums[k])")
                    sum += nums[k]
                }
                if sum >= target {
                    ans = min(ans, (j - i + 1))
                    break
                }
                print("")
            }
        }
        return ans == Int.max ? -1 : ans
    }
}

▼ 從 log 可以看到

// minSubArrayLen(7,  [2,3,1,2,4,3])

<i start:0>      
sum += 2

sum += 2
sum += 3

sum += 2
sum += 3
sum += 1

sum += 2
sum += 3
sum += 1
sum += 2
...                

可以看到一個現象,每一次要計算 sum 都會把前面已經算得再算一次

例如 2+32+3+1 時已經算過了,就不用再算一次

▼ 使用 Sliding Widnow 技巧的程式碼:

class Solution {
    func minSubArrayLen(_ target: Int, _ nums: [Int]) -> Int {
        var left = 0
        var subsum = 0
        var ans = Int.max
        for (right, _) in nums.enumerated() {                    
            subsum += nums[right]
            print("sum += \(nums[right])")            
            while subsum >= target && left <= right {
                print("sum -= \(nums[left]) (sum >= target(\(target)))")
                ans = min(ans,(right - left + 1))
                subsum -= nums[left]
                left += 1
            }
            print("")
        }
        return ans == Int.max ? -1 : ans
    }
}
// minSubArrayLen(7,  [2,3,1,2,4,3])

sum += 2

sum += 3

sum += 1

sum += 2
sum -= 2 (sum >= target(7))

sum += 4
sum -= 3 (sum >= target(7))
sum -= 1 (sum >= target(7))

sum += 3
sum -= 2 (sum >= target(7))
sum -= 4 (sum >= target(7))

由 log 可以明顯看出,一開始加 2 + 3 + 1 + 2 之後判斷大於 target 7 ,開始操作 slide window 的 left 跟 right 指摽來控制 sum

▼ 下圖可以清楚看出 2 + 3 + 1 + 2 之後的 steps 如何移動 window

發佈留言

發佈留言必須填寫的電子郵件地址不會公開。 必填欄位標示為 *