iOS中的Throttle(函數(shù)節(jié)流)與Debounce(函數(shù)防抖)

為什么需要Throttle和Debounce

Throttle和Debounce在前端開(kāi)發(fā)可能比較經(jīng)常用到,做iOS開(kāi)發(fā)可能很多人不知道這個(gè)這個(gè)概念,其實(shí)很開(kāi)發(fā)者在工作中或多或少都遇到過(guò),就像設(shè)計(jì)模式有很多種,開(kāi)發(fā)中用到了某種設(shè)計(jì)模式自己卻不知道,這篇文章我們就簡(jiǎn)單聊Throttle和Debounce。
開(kāi)發(fā)中我們都遇到頻率很高的事件(如搜索框的搜索)或者連續(xù)事件(如UIScrollView的contentOffset進(jìn)行某些計(jì)算),這個(gè)時(shí)候?yàn)榱诉M(jìn)行性能優(yōu)化就要用到Throttle和Debounce。在詳細(xì)說(shuō)這連個(gè)概念之前我們先弄清楚一件事就是觸發(fā)事件和執(zhí)行事件對(duì)應(yīng)的方法是不同的。舉個(gè)栗子,有個(gè)button,我們點(diǎn)擊是觸發(fā)了點(diǎn)擊事件和之后比如進(jìn)行網(wǎng)絡(luò)這個(gè)方法是不一樣的,Throttle和Debounce并不會(huì)限制你去觸發(fā)點(diǎn)擊事件,但是會(huì)控制之后的方法調(diào)用,這和我們?cè)O(shè)置一種機(jī)制,去設(shè)置button的isEnable的方式是不同的。

Debounce

當(dāng)事件觸發(fā)超過(guò)一段時(shí)間之后才會(huì)執(zhí)行方法,如果在這段時(shí)間之內(nèi)有又觸發(fā)了這個(gè)時(shí)間,則重新計(jì)算時(shí)間。
電梯的處理就和這個(gè)類(lèi)似,比如現(xiàn)在在4樓,有個(gè)人按了1樓的按鈕(事件),這個(gè)時(shí)候電梯會(huì)等一固定時(shí)間,如果沒(méi)人再按按鈕,則電梯開(kāi)始下降(對(duì)應(yīng)的方法),如果有人立馬又按了1樓按鈕,電梯就會(huì)重新計(jì)算時(shí)間。
我們看看在面對(duì)search問(wèn)題上可以怎么處理

第一版

class SearchViewController: UIViewController, UISearchBarDelegate {
    // We keep track of the pending work item as a property
    private var pendingRequestWorkItem: DispatchWorkItem?

    func searchBar(_ searchBar: UISearchBar, textDidChange searchText: String) {
        // Cancel the currently pending item
        pendingRequestWorkItem?.cancel()

        // Wrap our request in a work item
        let requestWorkItem = DispatchWorkItem { [weak self] in
            self?.resultsLoader.loadResults(forQuery: searchText)
        }

        // Save the new work item and execute it after 250 ms
        pendingRequestWorkItem = requestWorkItem
        DispatchQueue.main.asyncAfter(deadline: .now() + .milliseconds(250),
                                      execute: requestWorkItem)
    }
}

這里運(yùn)用了DispatchWorkItem,將請(qǐng)求放在代碼塊中,當(dāng)有一個(gè)請(qǐng)求來(lái)時(shí)我們可以輕易的取消請(qǐng)求。正如你上面看到的,使用DispatchWorkItem在Swift中實(shí)際上比使用Timer或者Operation要好得多,這要?dú)w功于尾隨的閉包語(yǔ)法,以及GCD如何導(dǎo)入Swift。 你不需要@objc標(biāo)記的方法,或#selector,它可以全部使用閉包完成。

第二版
但只是這樣肯定不行的,我們?cè)囍シ庋b一下好在其他地方也能同樣使用。下面我們看看參考文章里的一個(gè)寫(xiě)法,當(dāng)然還有用Timer實(shí)現(xiàn)的,讀者感興趣可以自己看看

typealias Debounce<T> = (_ : T) -> Void

func debounce<T>(interval: Int, queue: DispatchQueue, action: @escaping Debounce<T>) -> Debounce<T> {
    var lastFireTime = DispatchTime.now()
    let dispatchDelay = DispatchTimeInterval.milliseconds(interval)

    return { param in
        lastFireTime = DispatchTime.now()
        let dispatchTime: DispatchTime = DispatchTime.now() + dispatchDelay

        queue.asyncAfter(deadline: dispatchTime) {
            let when: DispatchTime = lastFireTime + dispatchDelay
            let now = DispatchTime.now()

            if now.rawValue >= when.rawValue {
                action(param)
            }
        }
    }
}

第三版

下面我們?cè)賹?duì)其進(jìn)行改進(jìn),一是使用DispatchWorkItem,二是使用DispatchSemaphore保證線(xiàn)程安全。

class Debouncer {
    public let label: String
    public let interval: DispatchTimeInterval
    fileprivate let queue: DispatchQueue
    fileprivate let semaphore: DispatchSemaphoreWrapper
    fileprivate var workItem: DispatchWorkItem?
    
    
    public init(label: String, interval: Float, qos: DispatchQoS = .userInteractive) {
        self.interval         = .milliseconds(Int(interval * 1000))
        self.label         = label
        self.queue = DispatchQueue(label: "com.farfetch.debouncer.internalqueue.\(label)", qos: qos)
        self.semaphore = DispatchSemaphoreWrapper(withValue: 1)
    }
    
    
    public func call(_ callback: @escaping (() -> ())) {
        
        self.semaphore.sync  { () -> () in
            
            
            self.workItem?.cancel()
            
            self.workItem = DispatchWorkItem {
                callback()
            }
            
            if let workItem = self.workItem {
                
                self.queue.asyncAfter(deadline: .now() + self.interval, execute: workItem)
            }
        }
    }
    
}


public struct DispatchSemaphoreWrapper {
    
    private let semaphore: DispatchSemaphore
    
    public init(withValue value: Int) {
        
        self.semaphore = DispatchSemaphore(value: value)
    }
    
    public func sync<R>(execute: () throws -> R) rethrows -> R {
        
        _ = semaphore.wait(timeout: DispatchTime.distantFuture)
        defer { semaphore.signal() }
        return try execute()
    }
}

Throttle

預(yù)先設(shè)定一個(gè)執(zhí)行周期,當(dāng)調(diào)用動(dòng)作大于等于執(zhí)行周期則執(zhí)行該動(dòng)作,然后進(jìn)入下一個(gè)新的時(shí)間周期
這有點(diǎn)像班車(chē)系統(tǒng)和這個(gè)類(lèi)似,比如一個(gè)班車(chē)每隔15分鐘發(fā)車(chē),有人來(lái)了就上車(chē),到了15分鐘就發(fā)車(chē),不管中間有多少乘客上車(chē)。

import UIKit
import Foundation
 
public class Throttler {
    
    private let queue: DispatchQueue = DispatchQueue.global(qos: .background)
    
    private var job: DispatchWorkItem = DispatchWorkItem(block: {})
    private var previousRun: Date = Date.distantPast
    private var maxInterval: Int
    fileprivate let semaphore: DispatchSemaphoreWrapper
    
    init(seconds: Int) {
        self.maxInterval = seconds
        self.semaphore = DispatchSemaphoreWrapper(withValue: 1)
    }
    
    
    func throttle(block: @escaping () -> ()) {
        
        self.semaphore.sync  { () -> () in
            job.cancel()
            job = DispatchWorkItem(){ [weak self] in
                self?.previousRun = Date()
                block()
            }
            let delay = Date.second(from: previousRun) > maxInterval ? 0 : maxInterval
            queue.asyncAfter(deadline: .now() + Double(delay), execute: job)
        }
        
    }
}
 
private extension Date {
    static func second(from referenceDate: Date) -> Int {
        return Int(Date().timeIntervalSince(referenceDate).rounded())
    }
}

示例

import UIKit
 
public class SearchBar: UISearchBar, UISearchBarDelegate {
    
    /// Throttle engine
    private var throttler: Throttler? = nil
    
    /// Throttling interval
    public var throttlingInterval: Double? = 0 {
        didSet {
            guard let interval = throttlingInterval else {
                self.throttler = nil
                return
            }
            self.throttler = Throttler(seconds: interval)
        }
    }
    
    /// Event received when cancel is pressed
    public var onCancel: (() -> (Void))? = nil
    
    /// Event received when a change into the search box is occurred
    public var onSearch: ((String) -> (Void))? = nil
    
    public override func awakeFromNib() {
        super.awakeFromNib()
        self.delegate = self
    }
    
    // Events for UISearchBarDelegate
    
    public func searchBarCancelButtonClicked(_ searchBar: UISearchBar) {
        self.onCancel?()
    }
    
    public func searchBarSearchButtonClicked(_ searchBar: UISearchBar) {
        self.onSearch?(self.text ?? "")
    }
    
    public func searchBar(_ searchBar: UISearchBar, textDidChange searchText: String) {
        guard let throttler = self.throttler else {
            self.onSearch?(searchText)
            return
        }
        throttler.throttle {
            DispatchQueue.main.async {
                self.onSearch?(self.text ?? "")
            }
        }
    }
    
}

思考

根據(jù)Debounce我們知道如果一直去觸發(fā)某個(gè)事件,那么就會(huì)造成一直無(wú)法調(diào)用相應(yīng)的方法,那么我們可以設(shè)置一個(gè)最大等待時(shí)間maxInterval,當(dāng)超過(guò)這個(gè)時(shí)間則執(zhí)行相應(yīng)的方法,避免一直等待。具體實(shí)施就不寫(xiě)了,讀者結(jié)合Debounce和Throttle可以自己去實(shí)現(xiàn),哈哈,這個(gè)有點(diǎn)像Debounce和Throttle的雜交品種。

參考文章

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,702評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,143評(píng)論 3 415
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 175,553評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 62,620評(píng)論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,416評(píng)論 6 405
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 54,940評(píng)論 1 321
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,024評(píng)論 3 440
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,170評(píng)論 0 287
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,709評(píng)論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,597評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,784評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,291評(píng)論 5 357
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,029評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,407評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,663評(píng)論 1 280
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,403評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,746評(píng)論 2 370

推薦閱讀更多精彩內(nèi)容

  • 1、通過(guò)CocoaPods安裝項(xiàng)目名稱(chēng)項(xiàng)目信息 AFNetworking網(wǎng)絡(luò)請(qǐng)求組件 FMDB本地?cái)?shù)據(jù)庫(kù)組件 SD...
    陽(yáng)明先生_X自主閱讀 16,000評(píng)論 3 119
  • 福利院有顆大榕樹(shù),樹(shù)下有把桃紅色的塑料椅,一位老奶奶坐在椅子上,雙手相扣,半張著嘴,瞇眼睡去。風(fēng),掠過(guò)她桃紅色的發(fā)...
    西山有井閱讀 204評(píng)論 0 1
  • 三分酒意釀成涼,且無(wú)妨,醉思量。莫道向誰(shuí),枉斷相思腸。即使天涯隨夢(mèng)海,如雁字,和鴻翔。 江山無(wú)限最成傷,念西窗,月...
    迷曳閱讀 762評(píng)論 5 3
  • 文/雨隨塵清 走過(guò)曲折的山路,踏過(guò)清涼的小溪,領(lǐng)略過(guò)晨起初生的朝陽(yáng),欣賞過(guò)朦朧的月色和漫天的星光。 車(chē)窗外的風(fēng)景,...
    清陋閱讀 297評(píng)論 14 17