欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

EvenLoop模型在iOS的RunLoop應(yīng)用示例

 更新時間:2022年07月20日 11:26:51   作者:向輝_  
這篇文章主要為大家介紹了EvenLoop模型在iOS的RunLoop應(yīng)用示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

引言

Runloop在iOS中是一個很重要的組成部分,對于任何單線程的UI模型都必須使用EvenLoop才可以連續(xù)處理不同的事件,而RunLoop就是EvenLoop模型在iOS中的實現(xiàn)。在前面的幾篇文章中,我已經(jīng)介紹了Runloop的底層原理等,這篇文章主要是從實際開發(fā)的角度,探討一下實際上在哪些場景下,我們可以去使用RunLoop。

線程?;?/h2>

在實際開發(fā)中,我們通常會遇到常駐線程的創(chuàng)建,比如說發(fā)送心跳包,這就可以在一個常駐線程來發(fā)送心跳包,而不干擾主線程的行為,再比如音頻處理,這也可以在一個常駐線程中來處理。以前在Objective-C中使用的AFNetworking 1.0就使用了RunLoop來進行線程的?;?。

var thread: Thread!
func createLiveThread() {
		thread = Thread.init(block: {
				let port = NSMachPort.init()
        RunLoop.current.add(port, forMode: .default)
        RunLoop.current.run()
		})
		thread.start()
}

值得注意的是RunLoop的mode中至少需要一個port/timer/observer,否則RunLoop只會執(zhí)行一次就退出了。

停止Runloop

離開RunLoop一共有兩種方法:其一是給RunLoop配置一個超時的時間,其二是主動通知RunLoop離開。Apple在文檔中是推薦第一種方式的,如果能直接定量的管理,這種方式當(dāng)然是最好的。

設(shè)置超時時間

然而實際中我們無法準確的去設(shè)置超時的時刻,比如在線程?;畹睦又校覀冃枰WC線程的RunLoop一直保持運行中,所以結(jié)束的時間是一個變量,而不是常量,要達到這個目標我們可以結(jié)合一下RunLoop提供的API,在開始的時候,設(shè)置RunLoop超時時間為無限,但是在結(jié)束時,設(shè)置RunLoop超時時間為當(dāng)前,這樣變相通過控制timeout的時間停止了RunLoop,具體代碼如下:

var thread: Thread?
var isStopped: Bool = false
func createLiveThread() {
		thread = Thread.init(block: { [weak self] in
				guard let self = self else { return }
				let port = NSMachPort.init()
        RunLoop.current.add(port, forMode: .default)
				while !self.isStopped {
		        RunLoop.current.run(mode: .default, before: Date.distantFuture)
        }
		})
		thread?.start()
}
func stop() {
		self.perform(#selector(self.stopThread), on: thread!, with: nil, waitUntilDone: false)
}
@objc func stopThread() {
		self.isStopped = true
		RunLoop.current.run(mode: .default, before: Date.init())
    self.thread = nil
}

直接停止

CoreFoundation提供了API:CFRunLoopStop() 但是這個方法只會停止當(dāng)前這次循環(huán)的RunLoop,并不會完全停止RunLoop。那么有沒有其它的策略呢?我們知道RunLoop的Mode中必須要至少有一個port/timer/observer才會工作,否則就會退出,而CF提供的API中正好有:

**public func CFRunLoopRemoveSource(_ rl: CFRunLoop!, _ source: CFRunLoopSource!, _ mode: CFRunLoopMode!)
public func CFRunLoopRemoveObserver(_ rl: CFRunLoop!, _ observer: CFRunLoopObserver!, _ mode: CFRunLoopMode!)
public func CFRunLoopRemoveTimer(_ rl: CFRunLoop!, _ timer: CFRunLoopTimer!, _ mode: CFRunLoopMode!)**

所以很自然的聯(lián)想到如果移除source/timer/observer, 那么這個方案可不可以停止RunLoop呢?

答案是否定的,這一點在Apple的官方文檔中有比較詳細的描述:

Although removing a run loop’s input sources and timers may also cause the run loop to exit, this is not a reliable way to stop a run loop. Some system routines add input sources to a run loop to handle needed events. Because your code might not be aware of these input sources, it would be unable to remove them, which would prevent the run loop from exiting.

簡而言之,就是你無法保證你移除的就是全部的source/timer/observer,因為系統(tǒng)可能會添加一些必要的source來處理事件,而這些source你是無法確保移除的。

延遲加載圖片

這是一個很常見的使用方式,因為我們在滑動scrollView/tableView/collectionView的過程,總會給cell設(shè)置圖片,但是直接給cell的imageView設(shè)置圖片的過程中,會涉及到圖片的解碼操作,這個就會占用CPU的計算資源,可能導(dǎo)致主線程發(fā)生卡頓,所以這里可以將這個操作,不放在trackingMode,而是放在defaultMode中,通過一種取巧的方式來解決可能的性能問題。

func setupImageView() {
		self.performSelector(onMainThread: #selector(self.setupImage), 
												 with: nil, 
												 waitUntilDone: false,
												 modes: [RunLoop.Mode.default.rawValue])
}
@objc func setupImage() {
		imageView.setImage()
}

卡頓監(jiān)測

目前來說,一共有三種卡頓監(jiān)測的方案,然而基本上每一種卡頓監(jiān)測的方案都和RunLoop是有關(guān)聯(lián)的。

CADisplayLink(FPS)

YYFPSLabel 采用的就是這個方案,F(xiàn)PS(Frames Per Second)代表每秒渲染的幀數(shù),一般來說,如果App的FPS保持50~60之間,用戶的體驗就是比較流暢的,但是Apple自從iPhone支持120HZ的高刷之后,它發(fā)明了一種ProMotion的動態(tài)屏幕刷新率的技術(shù),這種方式基本就不能使用了,但是這里依舊提供已作參考。

這里值得注意的技術(shù)細節(jié)是使用了NSObject來做方法的轉(zhuǎn)發(fā),在OC中可以使用NSProxy來做消息的轉(zhuǎn)發(fā),效率更高。

// 抽象的超類,用來充當(dāng)其它對象的一個替身
// Timer/CADisplayLink可以使用NSProxy做消息轉(zhuǎn)發(fā),可以避免循環(huán)引用
// swift中我們是沒發(fā)使用NSInvocation的,所以我們直接使用NSobject來做消息轉(zhuǎn)發(fā)
class WeakProxy: NSObject {
    private weak var target: NSObjectProtocol?
    init(target: NSObjectProtocol) {
        self.target = target
        super.init()
    }
    override func responds(to aSelector: Selector!) -> Bool {
        return (target?.responds(to: aSelector) ?? false) || super.responds(to: aSelector)
    }
    override func forwardingTarget(for aSelector: Selector!) -> Any? {
        return target
    }
}
class FPSLabel: UILabel {
    var link: CADisplayLink!
    var count: Int = 0
    var lastTime: TimeInterval = 0.0
    fileprivate let defaultSize = CGSize.init(width: 80, height: 20)
    override init(frame: CGRect) {
        super.init(frame: frame)
        if frame.size.width == 0 || frame.size.height == 0 {
            self.frame.size = defaultSize
        }
        layer.cornerRadius = 5.0
        clipsToBounds = true
        textAlignment = .center
        isUserInteractionEnabled = false
        backgroundColor = UIColor.white.withAlphaComponent(0.7)
        link = CADisplayLink.init(target: WeakProxy.init(target: self), selector: #selector(FPSLabel.tick(link:)))
        link.add(to: RunLoop.main, forMode: .common)
    }
    required init?(coder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
    deinit {
        link.invalidate()
    }
    @objc func tick(link: CADisplayLink) {
        guard lastTime != 0 else {
            lastTime = link.timestamp
            return
        }
        count += 1
        let timeDuration = link.timestamp - lastTime
        // 1、設(shè)置刷新的時間: 這里是設(shè)置為1秒(即每秒刷新)
        guard timeDuration >= 1.0 else { return }
        // 2、計算當(dāng)前的FPS
        let fps = Double(count)/timeDuration
        count = 0
        lastTime = link.timestamp
        // 3、開始設(shè)置FPS了
        let progress = fps/60.0
        let color = UIColor(hue: CGFloat(0.27 * (progress - 0.2)), saturation: 1, brightness: 0.9, alpha: 1)
        self.text = "\(Int(round(fps))) FPS"
        self.textColor = color
    }
}

子線程Ping

這種方法是創(chuàng)建了一個子線程,通過GCD給主線程添加異步任務(wù):修改是否超時的參數(shù),然后讓子線程休眠一段時間,如果休眠的時間結(jié)束之后,超時參數(shù)未修改,那說明給主線程的任務(wù)并沒有執(zhí)行,那么這就說明主線程的上一個任務(wù)還沒有做完,那就說明卡頓了,這種方式其實和RunLoop沒有太多的關(guān)聯(lián),它不依賴RunLoop的狀態(tài)。在ANREye中是采用子線程Ping的方式來監(jiān)測卡頓的。

同時為了讓這些操作是同步的,這里使用了信號量。

class PingMonitor {
    static let timeoutInterval: TimeInterval = 0.2
    static let queueIdentifier: String = "com.queue.PingMonitor"
    private var queue: DispatchQueue = DispatchQueue.init(label: queueIdentifier)
    private var isMonitor: Bool = false
    private var semphore: DispatchSemaphore = DispatchSemaphore.init(value: 0)
    func startMonitor() {
        guard isMonitor == false else { return }
        isMonitor = true
        queue.async {
            while self.isMonitor {
                var timeout = true
                DispatchQueue.main.async {
                    timeout = false
                    self.semphore.signal()
                }
                Thread.sleep(forTimeInterval:PingMonitor.timeoutInterval)
                // 說明等了timeoutInterval之后,主線程依然沒有執(zhí)行派發(fā)的任務(wù),這里就認為它是處于卡頓的
                if timeout == true {
                    //TODO: 這里需要取出崩潰方法棧中的符號來判斷為什么出現(xiàn)了卡頓
                    // 可以使用微軟的框架:PLCrashReporter
                }
                self.semphore.wait()
            }
        }
    }
}

這個方法在正常情況下會每隔一段時間讓主線程執(zhí)行GCD派發(fā)的任務(wù),會造成部分資源的浪費,而且它是一種主動的去Ping主線程,并不能很及時的發(fā)現(xiàn)卡頓問題,所以這種方法會有一些缺點。

實時監(jiān)控

而我們知道,主線程中任務(wù)都是通過RunLoop來管理執(zhí)行的,所以我們可以通過監(jiān)聽RunLoop的狀態(tài)來知道是否會出現(xiàn)卡頓的情況,一般來說,我們會監(jiān)測兩種狀態(tài):第一種是kCFRunLoopAfterWaiting 的狀態(tài),第二種是kCFRunLoopBeforeSource的狀態(tài)。為什么是兩種狀態(tài)呢?

首先看第一種狀態(tài)kCFRunLoopAfterWaiting ,它會在RunLoop被喚醒之后回調(diào)這種狀態(tài),然后根據(jù)被喚醒的端口來處理不同的任務(wù),如果處理任務(wù)的過程中耗時過長,那么下一次檢查的時候,它依然是這個狀態(tài),這個時候就可以說明它卡在了這個狀態(tài)了,然后可以通過一些策略來提取出方法棧,來判斷卡頓的代碼。同理,第二種狀態(tài)也是一樣的,說明一直處于kCFRunLoopBeforeSource 狀態(tài),而沒有進入下一狀態(tài)(即休眠),也發(fā)生了卡頓。

class RunLoopMonitor {
    private init() {}
    static let shared: RunLoopMonitor = RunLoopMonitor.init()
    var timeoutCount = 0
    var runloopObserver: CFRunLoopObserver?
    var runLoopActivity: CFRunLoopActivity?
    var dispatchSemaphore: DispatchSemaphore?
    // 原理:進入睡眠前方法的執(zhí)行時間過長導(dǎo)致無法進入睡眠,或者線程喚醒之后,一直沒進入下一步
    func beginMonitor() {
        let uptr = Unmanaged.passRetained(self).toOpaque()
        let vptr = UnsafeMutableRawPointer(uptr)
        var context = CFRunLoopObserverContext.init(version: 0, info: vptr, retain: nil, release: nil, copyDescription: nil)
        runloopObserver = CFRunLoopObserverCreate(kCFAllocatorDefault,
                                                  CFRunLoopActivity.allActivities.rawValue,
                                                  true,
                                                  0,
                                                  observerCallBack(),
                                                  &context)
        CFRunLoopAddObserver(CFRunLoopGetMain(), runloopObserver, .commonModes)
        // 初始化的信號量為0
        dispatchSemaphore = DispatchSemaphore.init(value: 0)
        DispatchQueue.global().async {
            while true {
                // 方案一:可以通過設(shè)置單次超時時間來判斷 比如250毫秒
								// 方案二:可以通過設(shè)置連續(xù)多次超時就是卡頓 戴銘在GCDFetchFeed中認為連續(xù)三次超時80秒就是卡頓
                let st = self.dispatchSemaphore?.wait(timeout: DispatchTime.now() + .milliseconds(80))
                if st == .timedOut {
                    guard self.runloopObserver != nil else {
                        self.dispatchSemaphore = nil
                        self.runLoopActivity = nil
												self.timeoutCount = 0
                        return
                    }
                    if self.runLoopActivity == .afterWaiting || self.runLoopActivity == .beforeSources {
												self.timeoutCount += 1
                        if self.timeoutCount < 3 { continue }
                        DispatchQueue.global().async {
                            let config = PLCrashReporterConfig.init(signalHandlerType: .BSD, symbolicationStrategy: .all)
                            guard let crashReporter = PLCrashReporter.init(configuration: config) else { return }
                            let data = crashReporter.generateLiveReport()
                            do {
                                let reporter = try PLCrashReport.init(data: data)
                                let report = PLCrashReportTextFormatter.stringValue(for: reporter, with: PLCrashReportTextFormatiOS) ?? ""
                                NSLog("------------卡頓時方法棧:\n \(report)\n")
                            } catch _ {
                                NSLog("解析crash data錯誤")
                            }
                        }
                    }
                }
            }
        }
    }
    func end() {
        guard let _ = runloopObserver else { return }
        CFRunLoopRemoveObserver(CFRunLoopGetMain(), runloopObserver, .commonModes)
        runloopObserver = nil
    }
    private func observerCallBack() -> CFRunLoopObserverCallBack {
        return { (observer, activity, context) in
            let weakself = Unmanaged<RunLoopMonitor>.fromOpaque(context!).takeUnretainedValue()
            weakself.runLoopActivity = activity
            weakself.dispatchSemaphore?.signal()
        }
    }
}

Crash防護

Crash防護是一個很有意思的點,處于應(yīng)用層的APP,在執(zhí)行了某些不被操作系統(tǒng)允許的操作之后會觸發(fā)操作系統(tǒng)拋出異常信號,但是因為沒有處理這些異常從而被系操作系統(tǒng)殺掉的線程,比如常見的閃退。這里不對Crash做詳細的描述,我會在下一個模塊來描述iOS中的異常。要明確的是,有些場景下,是希望可以捕獲到系統(tǒng)拋出的異常,然后將App從錯誤中恢復(fù),重新啟動,而不是被殺死。而對應(yīng)在代碼中,我們需要去手動的重啟主線程,已達到繼續(xù)運行App的目的。

let runloop = CFRunLoopGetCurrent()
guard let allModes = CFRunLoopCopyAllModes(runloop) as? [CFRunLoopMode] else {
    return
}
 while true {
	  for mode in allModes {
        CFRunLoopRunInMode(mode, 0.001, false)
    }
 }

CFRunLoopRunInMode(mode, 0.001, false) 因為無法確定RunLoop到底是怎樣啟動的,所以采用了這種方式來啟動RunLoop的每一個Mode,也算是一種替代方案了。因為CFRunLoopRunInMode 在運行的時候本身就是一個循環(huán)并不會退出,所以while循環(huán)不會一直執(zhí)行,只是在mode退出之后,while循環(huán)遍歷需要執(zhí)行的mode,直到繼續(xù)在一個mode中常駐。

這里只是重啟RunLoop,其實在Crash防護里最重要的還是要監(jiān)測到何時發(fā)送崩潰,捕獲系統(tǒng)的exception信息,以及singal信息等等,捕獲到之后再對當(dāng)前線程的方法棧進行分析,定位為crash的成因。

Matrix框架

接下來我們具體看一下RunLoop在Matrix框架中的運用。Matrix是騰訊開源的一款用于性能監(jiān)測的框架,在這個框架中有一款插件**WCFPSMonitorPlugin:**這是一款FPS監(jiān)控工具,當(dāng)用戶滑動界面時,記錄主線程的調(diào)用棧。它的源碼中和我們上述提到的通過CADisplayLink來來監(jiān)測卡頓的方案的原理是一樣的:

- (void)startDisplayLink:(NSString *)scene {
    FPSInfo(@"startDisplayLink");
    m_displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(onFrameCallback:)];
    [m_displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSRunLoopCommonModes];
		...
}
- (void)onFrameCallback:(id)sender {
    // 當(dāng)前時間: 單位為秒
    double nowTime = CFAbsoluteTimeGetCurrent();
    // 將單位轉(zhuǎn)化為毫秒
    double diff = (nowTime - m_lastTime) * 1000;
		// 1、如果時間間隔超過最大的幀間隔:那么此次屏幕刷新方法超時
    if (diff > self.pluginConfig.maxFrameInterval) {
        m_currRecorder.dumpTimeTotal += diff;
        m_dropTime += self.pluginConfig.maxFrameInterval * pow(diff / self.pluginConfig.maxFrameInterval, self.pluginConfig.powFactor);
        // 總超時時間超過閾值:展示超時信息
        if (m_currRecorder.dumpTimeTotal > self.pluginConfig.dumpInterval * self.pluginConfig.dumpMaxCount) {
            FPSInfo(@"diff %lf exceed, begin: %lf, end: %lf, scene: %@, you can see more detail in record id: %d",
                    m_currRecorder.dumpTimeTotal,
                    m_currRecorder.dumpTimeBegin,
                    m_currRecorder.dumpTimeBegin + m_currRecorder.dumpTimeTotal / 1000.0,
                    m_scene,
                    m_currRecorder.recordID);
						...... 
        }
		// 2、如果時間間隔沒有最大的幀間隔:那么此次屏幕刷新方法不超時
    } else {
        // 總超時時間超過閾值:展示超時信息
        if (m_currRecorder.dumpTimeTotal > self.pluginConfig.maxDumpTimestamp) {
            FPSInfo(@"diff %lf exceed, begin: %lf, end: %lf, scene: %@, you can see more detail in record id: %d",
                    m_currRecorder.dumpTimeTotal,
                    m_currRecorder.dumpTimeBegin,
                    m_currRecorder.dumpTimeBegin + m_currRecorder.dumpTimeTotal / 1000.0,
                    m_scene,
                    m_currRecorder.recordID);
						....
				// 總超時時間不超過閾值:將時間歸0 重新計數(shù)
        } else {
            m_currRecorder.dumpTimeTotal = 0;
            m_currRecorder.dumpTimeBegin = nowTime + 0.0001;
        }
    }
    m_lastTime = nowTime;
}

它通過次數(shù)以及兩次之間允許的時間間隔作為閾值,超過閾值就記錄,沒超過閾值就歸0重新計數(shù)。當(dāng)然這個框架也不僅僅是作為一個簡單的卡頓監(jiān)測來使用的,還有很多性能監(jiān)測的功能以供平時開發(fā)的時候來使用:包括對崩潰時方法棧的分析等等。

總結(jié)

本篇文章我從線程?;铋_始介紹了RunLoop在實際開發(fā)中的使用,然后主要是介紹了卡頓監(jiān)測和Crash防護中的高階使用,當(dāng)然,RunLoop的運用遠不止這些,如果有更多更好的使用,希望大家可以留言交流。

以上就是EvenLoop模型在iOS的RunLoop應(yīng)用示例的詳細內(nèi)容,更多關(guān)于ios EvenLoop模型RunLoop的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Swift 4中一些實用的數(shù)組技巧小結(jié)

    Swift 4中一些實用的數(shù)組技巧小結(jié)

    這篇文章主要給大家分享了關(guān)于Swift 4中一些實用的數(shù)組技巧,文中通過示例代碼介紹的介紹的非常詳細,對大家學(xué)習(xí)或者使用swift具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2018-03-03
  • Swift網(wǎng)絡(luò)請求庫Alamofire使用詳解

    Swift網(wǎng)絡(luò)請求庫Alamofire使用詳解

    這篇文章主要為大家詳細介紹了Swift網(wǎng)絡(luò)請求庫Alamofire的使用方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • 超全面的Swift編碼規(guī)范(推薦)

    超全面的Swift編碼規(guī)范(推薦)

    這篇文章主要給大家介紹了關(guān)于Swift編碼規(guī)范的相關(guān)資料,文中介紹的非常詳細,對大家開發(fā)swift具有一定的參考價值,需要的朋友可以參考學(xué)習(xí),下面來一起看看吧。
    2017-03-03
  • Swift設(shè)計思想Result<T>與Result<T,?E:?Error>類型解析

    Swift設(shè)計思想Result<T>與Result<T,?E:?Error>類型解析

    這篇文章主要為大家介紹了Swift設(shè)計思想Result<T>與Result<T,?E:?Error>的類型示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-11-11
  • Swift實現(xiàn)表格視圖單元格單選(1)

    Swift實現(xiàn)表格視圖單元格單選(1)

    這篇文章主要為大家詳細介紹了Swift實現(xiàn)表格視圖單元格單選,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • 深入解析Swift中switch語句對case的數(shù)據(jù)類型匹配的支持

    深入解析Swift中switch語句對case的數(shù)據(jù)類型匹配的支持

    這篇文章主要介紹了Swift中switch語句對case的數(shù)據(jù)類型匹配的支持,Swift中switch...case語句支持多種數(shù)據(jù)類型的匹配判斷,十分強大,需要的朋友可以參考下
    2016-04-04
  • Swift算法實現(xiàn)字符串轉(zhuǎn)數(shù)字的方法示例

    Swift算法實現(xiàn)字符串轉(zhuǎn)數(shù)字的方法示例

    最近學(xué)完了swift想著實踐下,就通過一些簡單的算法進行學(xué)習(xí)研究,下面這篇文章主要介紹了Swift算法實現(xiàn)字符串轉(zhuǎn)數(shù)字的方法,需要的朋友可以參考借鑒,下面來一起看看吧。
    2017-03-03
  • Swift map和filter函數(shù)原型基礎(chǔ)示例

    Swift map和filter函數(shù)原型基礎(chǔ)示例

    這篇文章主要為大家介紹了Swift map和filter函數(shù)原型基礎(chǔ)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-07-07
  • RxSwift學(xué)習(xí)教程之類型對象Subject詳解

    RxSwift學(xué)習(xí)教程之類型對象Subject詳解

    這篇文章主要給大家介紹了關(guān)于RxSwift學(xué)習(xí)教程之類型對象Subject的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起看看吧。
    2017-09-09
  • 升級到Swift 4.0可能遇到的坑總結(jié)

    升級到Swift 4.0可能遇到的坑總結(jié)

    這篇文章主要給大家介紹了關(guān)于升級到Swift 4.0可能遇到的坑的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家學(xué)習(xí)或者使用swift4具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-11-11

最新評論