日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) >

APP启动速度是门面,如何做到极致优化?

發(fā)布時(shí)間:2025/3/21 39 豆豆
生活随笔 收集整理的這篇文章主要介紹了 APP启动速度是门面,如何做到极致优化? 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

大家好,我是極客時(shí)間專欄《iOS 開(kāi)發(fā)高手課》的作者戴銘。之前我在專欄中跟大家梳理過(guò)成為一名開(kāi)發(fā)高手所必備的知識(shí)體系,感興趣的同學(xué)可以直接去APP里免費(fèi)看看。今天想和大家一起分享一些干貨,App 啟動(dòng)速度怎么做優(yōu)化與監(jiān)控?

在文章開(kāi)始前,我們先設(shè)想這么一個(gè)場(chǎng)景:假設(shè)你在排隊(duì)結(jié)賬時(shí),掏出手機(jī)打開(kāi)App甲準(zhǔn)備掃碼支付,結(jié)果半天進(jìn)不去,后面排隊(duì)的人給你壓力夠大吧。然后,你又打開(kāi)App乙,秒進(jìn),支付完成。試想一下,以后再支付時(shí)你會(huì)選擇哪個(gè)App呢。

不難想象,在提供的功能和服務(wù)相似的情況下,一款A(yù)pp的啟動(dòng)速度,不單單是用戶體驗(yàn)的事情,往往還決定了它能否獲取更多的用戶。這就好像陌生人第一次碰面,第一感覺(jué)往往決定了他們接下來(lái)是否會(huì)繼續(xù)交往。

由此可見(jiàn),啟動(dòng)速度的優(yōu)化必然就是App開(kāi)發(fā)過(guò)程中,不可或缺的一個(gè)環(huán)節(jié)。接下來(lái),我就先和你一起分析下App在啟動(dòng)時(shí)都做了哪些事兒。

App 啟動(dòng)時(shí)都干些了什么事兒?

一般情況下,App的啟動(dòng)分為冷啟動(dòng)和熱啟動(dòng)。

  • 冷啟動(dòng)是指, App 點(diǎn)擊啟動(dòng)前,它的進(jìn)程不在系統(tǒng)里,需要系統(tǒng)新創(chuàng)建一個(gè)進(jìn)程分配給它啟動(dòng)的情況。這是一次完整的啟動(dòng)過(guò)程。
  • 熱啟動(dòng)是指 ,App 在冷啟動(dòng)后用戶將 App 退后臺(tái),在 App 的進(jìn)程還在系統(tǒng)里的情況下,用戶重新啟動(dòng)進(jìn)入 App 的過(guò)程,這個(gè)過(guò)程做的事情非常少。

所以,今天這篇文章,我們就只展開(kāi)講App冷啟動(dòng)的優(yōu)化。

用戶能感知到的啟動(dòng)慢,其實(shí)都發(fā)生在主線程上。而主線程慢的原因有很多,比如在主線程上執(zhí)行了大文件讀寫(xiě)操作、在渲染周期中執(zhí)行了大量計(jì)算等。但是,有時(shí)你會(huì)發(fā)現(xiàn)即使你把首屏顯示之前的這些主線程的耗時(shí)問(wèn)題都解決了,還是比競(jìng)品啟動(dòng)得慢。

那么,究竟如何才能把啟動(dòng)時(shí)的所有耗時(shí)都找出來(lái)呢?解決這個(gè)問(wèn)題,你首先需要弄清楚 App在啟動(dòng)時(shí)都干了哪些事兒。

一般而言,App的啟動(dòng)時(shí)間,指的是從用戶點(diǎn)擊App開(kāi)始,到用戶看到第一個(gè)界面之間的時(shí)間。總結(jié)來(lái)說(shuō),App的啟動(dòng)主要包括三個(gè)階段:

  • main() 函數(shù)執(zhí)行前;
  • main() 函數(shù)執(zhí)行后;
  • 首屏渲染完成后。

整個(gè)啟動(dòng)過(guò)程的示意圖,如下所示:

圖1 App的整個(gè)啟動(dòng)過(guò)程

main() 函數(shù)執(zhí)行前

在 main() 函數(shù)執(zhí)行前,系統(tǒng)主要會(huì)做下面幾件事情:

  • 加載可執(zhí)行文件(,也就是App的.o文件的集合);
  • 加載動(dòng)態(tài)鏈接庫(kù),進(jìn)行 rebase 指針調(diào)整和 bind 符號(hào)綁定;
  • Objc 運(yùn)行時(shí)的初始處理,包括 Objc 相關(guān)類的注冊(cè)、category 注冊(cè)、selector 唯一性檢查等;
  • 初始化,包括了執(zhí)行 +load() 方法、attributeconstructor 修飾的函數(shù)的調(diào)用、創(chuàng)建 C++ 靜態(tài)全局變量。

相應(yīng)地,這個(gè)階段對(duì)于啟動(dòng)速度優(yōu)化來(lái)說(shuō),可以做的事情包括:

  • 減少動(dòng)態(tài)庫(kù)加載。每個(gè)庫(kù)本身都有依賴關(guān)系,蘋果公司建議使用更少的動(dòng)態(tài)庫(kù),并且建議在使用動(dòng)態(tài)庫(kù)的數(shù)量較多時(shí),盡量將多個(gè)動(dòng)態(tài)庫(kù)進(jìn)行合并。數(shù)量上,蘋果公司最多可以支持6個(gè)非系統(tǒng)動(dòng)態(tài)庫(kù)合并為一個(gè)。
  • 減少加載啟動(dòng)后不會(huì)去使用的類或者方法。
  • +load() 方法里的內(nèi)容可以放到首屏渲染完成后再執(zhí)行,或使用 +initialize() 方法替換掉。因?yàn)?#xff0c;在一個(gè) +load() 方法里,進(jìn)行運(yùn)行時(shí)方法替換操作會(huì)帶來(lái) 4 毫秒的消耗。不要小看這4毫秒,積少成多,執(zhí)行+load() 方法對(duì)啟動(dòng)速度的影響會(huì)越來(lái)越大。
  • 控制C++ 全局變量的數(shù)量。

main() 函數(shù)執(zhí)行后

main() 函數(shù)執(zhí)行后的階段,指的是從main()函數(shù)執(zhí)行開(kāi)始,到appDelegate 的 didFinishLaunchingWithOptions 方法里首屏渲染相關(guān)方法執(zhí)行完成。

首頁(yè)的業(yè)務(wù)代碼都是要在這個(gè)階段,也就是首屏渲染前執(zhí)行的,主要包括了:

  • 首屏初始化所需配置文件的讀寫(xiě)操作;
  • 首屏列表大數(shù)據(jù)的讀取;
  • 首屏渲染的大量計(jì)算等。

很多時(shí)候,開(kāi)發(fā)者會(huì)把各種初始化工作都放到這個(gè)階段執(zhí)行,導(dǎo)致渲染完成滯后。更加優(yōu)化的開(kāi)發(fā)方式,應(yīng)該是從功能上梳理出哪些是首屏渲染必要的初始化功能,哪些是 App 啟動(dòng)必要的初始化功能,而哪些是只需要在對(duì)應(yīng)功能開(kāi)始使用時(shí)才需要初始化的。梳理完之后,將這些初始化功能分別放到合適的階段進(jìn)行。

首屏渲染完成后

首屏渲染后的這個(gè)階段,主要完成的是,非首屏其他業(yè)務(wù)服務(wù)模塊的初始化、監(jiān)聽(tīng)的注冊(cè)、配置文件的讀取等。從函數(shù)上來(lái)看,這個(gè)階段指的就是截止到 didFinishLaunchingWithOptions 方法作用域內(nèi)執(zhí)行首屏渲染之后的所有方法執(zhí)行完成。簡(jiǎn)單說(shuō)的話,這個(gè)階段就是從渲染完成時(shí)開(kāi)始,到 didFinishLaunchingWithOptions 方法作用域結(jié)束時(shí)結(jié)束。

這個(gè)階段用戶已經(jīng)能夠看到 App 的首頁(yè)信息了,所以優(yōu)化的優(yōu)先級(jí)排在最后。但是,那些會(huì)卡住主線程的方法還是需要最優(yōu)先處理的,不然還是會(huì)影響到用戶后面的交互操作。

明白了App啟動(dòng)階段需要完成的工作后,我們就可以有的放矢地進(jìn)行啟動(dòng)速度的優(yōu)化了。這些優(yōu)化,包括了功能級(jí)別和方法級(jí)別的啟動(dòng)優(yōu)化。接下來(lái),我們就從這兩個(gè)角度展開(kāi)看看。

我們先來(lái)看看功能級(jí)別的啟動(dòng)優(yōu)化

我想,你所在的團(tuán)隊(duì)一定面臨過(guò)啟動(dòng)階段的代碼功能堆積、無(wú)規(guī)范、難維護(hù)的問(wèn)題吧。在 App 項(xiàng)目開(kāi)發(fā)初期,開(kāi)發(fā)人員不多、代碼量也沒(méi)那么大時(shí),這種情況比較少見(jiàn)。但到了后期 ,App 業(yè)務(wù)規(guī)模擴(kuò)大,團(tuán)隊(duì)人員水平參差不齊,各種代碼問(wèn)題就會(huì)爆發(fā)出來(lái),終歸需要來(lái)次全面治理。

而全面治理過(guò)程中的手段、方法和碰到的問(wèn)題,對(duì)于后面的規(guī)范制定以及啟動(dòng)速度監(jiān)控都有著重要的意義。那么,我們要怎樣從功能級(jí)別來(lái)進(jìn)行全面的啟動(dòng)優(yōu)化治理呢?

功能級(jí)別的啟動(dòng)優(yōu)化,就是要從main() 函數(shù)執(zhí)行后這個(gè)階段下手。

優(yōu)化的思路是: main() 函數(shù)開(kāi)始執(zhí)行后到首屏渲染完成前只處理首屏相關(guān)的業(yè)務(wù),其他非首屏業(yè)務(wù)的初始化、監(jiān)聽(tīng)注冊(cè)、配置文件讀取等都放到首屏渲染完成后去做。這里有一張功能級(jí)別的啟動(dòng)優(yōu)化示意圖,如下圖所示:

圖2 功能級(jí)別的啟動(dòng)優(yōu)化示意圖

接下來(lái),我們?cè)倏纯捶椒?jí)別的啟動(dòng)優(yōu)化

經(jīng)過(guò)功能級(jí)別的啟動(dòng)優(yōu)化,也就是將非首屏業(yè)務(wù)所需的功能滯后以后,從用戶點(diǎn)擊 App 到看到首屏的時(shí)間將會(huì)有很大程度的縮短,也就達(dá)到了優(yōu)化App啟動(dòng)速度的目的。

在這之后,我們需要進(jìn)一步做的,是檢查首屏渲染完成前主線程上有哪些耗時(shí)方法,將沒(méi)必要的耗時(shí)方法滯后或者異步執(zhí)行。通常情況下,耗時(shí)較長(zhǎng)的方法主要發(fā)生在計(jì)算大量數(shù)據(jù)的情況下,具體的表現(xiàn)就是加載、編輯、存儲(chǔ)圖片和文件等資源。

那么,你覺(jué)得是不是只需要優(yōu)化對(duì)資源的操作就可以了呢?

當(dāng)然不是。就像 +load() 方法,一個(gè)耗時(shí)4毫秒,100個(gè)就是400毫秒,這種耗時(shí)用戶也是能明顯感知到的。

比如,我以前使用的 ReactiveCocoa框架(這是一個(gè) iOS 上的響應(yīng)式編程框架),每創(chuàng)建一個(gè)信號(hào)都有6毫秒的耗時(shí)。這樣,稍不注意各種信號(hào)的創(chuàng)建就都被放在了首屏渲染完成前,進(jìn)而導(dǎo)致App的啟動(dòng)速度大幅變慢。

類似這樣單個(gè)方法耗時(shí)不多,但是由于堆積導(dǎo)致App啟動(dòng)速度大幅變慢的方法數(shù)不勝數(shù)。所以,你需要一個(gè)能夠?qū)?dòng)方法耗時(shí)進(jìn)行全面、精確檢查的手段。

那么問(wèn)題來(lái)了,有哪些監(jiān)控手段?這些監(jiān)控手段各有什么優(yōu)缺點(diǎn)?你又該如何選擇呢?

目前來(lái)看,對(duì)App啟動(dòng)速度的監(jiān)控,主要有兩種手段。

第一種方法是,定時(shí)抓取主線程上的方法調(diào)用堆棧,計(jì)算一段時(shí)間里各個(gè)方法的耗時(shí)。Xcode 工具套件里自帶的 Time Profiler ,采用的就是這種方式。

這種方式的優(yōu)點(diǎn)是,開(kāi)發(fā)類似工具成本不高,能夠快速開(kāi)發(fā)后集成到你的 App 中,以便在真實(shí)環(huán)境中進(jìn)行檢查。

說(shuō)到定時(shí)抓取,就會(huì)涉及到定時(shí)間隔的長(zhǎng)短問(wèn)題。

  • 定時(shí)間隔設(shè)置得長(zhǎng)了,會(huì)漏掉一些方法,從而導(dǎo)致檢查出來(lái)的耗時(shí)不精確;
  • 而定時(shí)間隔設(shè)置得短了,抓取堆棧這個(gè)方法本身調(diào)用過(guò)多也會(huì)影響整體耗時(shí),導(dǎo)致結(jié)果不準(zhǔn)確。

這個(gè)定時(shí)間隔如果小于所有方法執(zhí)行的時(shí)間(比如 0.002秒),那么基本就能監(jiān)控到所有方法。但這樣做的話,整體的耗時(shí)時(shí)間就不夠準(zhǔn)確。一般將這個(gè)定時(shí)間隔設(shè)置為0.01秒。這樣設(shè)置,對(duì)整體耗時(shí)的影響小,不過(guò)很多方法耗時(shí)就不精確了。但因?yàn)檎w耗時(shí)的數(shù)據(jù)更加重要些,單個(gè)方法耗時(shí)精度不高也是可以接受的,所以這個(gè)設(shè)置也是沒(méi)問(wèn)題的。

總結(jié)來(lái)說(shuō),定時(shí)抓取主線程調(diào)用棧的方式雖然精準(zhǔn)度不夠高,但也是夠用的。

第二種方法是,對(duì) objc_msgSend 方法進(jìn)行 hook 來(lái)掌握所有方法的執(zhí)行耗時(shí)。

hook 方法的意思是,在原方法開(kāi)始執(zhí)行時(shí)換成執(zhí)行其他你指定的方法,或者在原有方法執(zhí)行前后執(zhí)行你指定的方法,來(lái)達(dá)到掌握和改變指定方法的目的。

hook objc_msgSend 這種方式的優(yōu)點(diǎn)是非常精確,而缺點(diǎn)是只能針對(duì) Objective-C 的方法。當(dāng)然,對(duì)于 c 方法和 block 也不是沒(méi)有辦法,你可以使用 libffi 的 ffi_call 來(lái)達(dá)成 hook,但缺點(diǎn)就是編寫(xiě)維護(hù)相關(guān)工具門檻高。

關(guān)于,libffi 相關(guān)的內(nèi)容,我會(huì)在后面的第35篇文章“l(fā)ibffi:動(dòng)態(tài)調(diào)用和定義 C 函數(shù)”里和你詳細(xì)說(shuō)明。

綜上,如果對(duì)于檢查結(jié)果精準(zhǔn)度要求高的話,我比較推薦你使用 hook objc_msgSend 方式來(lái)檢查啟動(dòng)方法的執(zhí)行耗時(shí)。

如何做一個(gè)方法級(jí)別啟動(dòng)耗時(shí)檢查工具來(lái)輔助分析和監(jiān)控?

使用 hook objc_msgSend 方式來(lái)檢查啟動(dòng)方法的執(zhí)行耗時(shí)時(shí),我們需要實(shí)現(xiàn)一個(gè)稱手的啟動(dòng)時(shí)間檢查工具。那么,我們應(yīng)該如何實(shí)現(xiàn)這個(gè)工具呢?

現(xiàn)在,我就一步一步地和你說(shuō)說(shuō)具體怎么做。

首先,你要了解為什么 hook 了 objc_msgSend 方法,就可以 hook 全部 Objective-C 的方法?

Objective-C 里每個(gè)對(duì)象都會(huì)指向一個(gè)類,每個(gè)類都會(huì)有一個(gè)方法列表,方法列表里的每個(gè)方法都是由 selector、函數(shù)指針和 metadata 組成的。

objc_msgSend 方法干的活兒,就是在運(yùn)行時(shí)根據(jù)對(duì)象和方法的selector 去找到對(duì)應(yīng)的函數(shù)指針,然后執(zhí)行。也就是說(shuō),objc_msgSend 是 Objective-C 里方法執(zhí)行的必經(jīng)之路,能夠控制所有的 Objective-C 的方法。

objc_msgSend 本身是用匯編語(yǔ)言寫(xiě)的,這樣做的原因主要有兩個(gè):

  • 一個(gè)原因是,objc_msgSend 的調(diào)用頻次最高,在它上面進(jìn)行的性能優(yōu)化能夠提升整個(gè) App 生命周期的性能。而匯編語(yǔ)言在性能優(yōu)化上屬于原子級(jí)優(yōu)化,能夠把優(yōu)化做到極致。所以,這種投入產(chǎn)出比無(wú)疑是最大的。
  • 另一個(gè)原因是,其他語(yǔ)言難以實(shí)現(xiàn)未知參數(shù)跳轉(zhuǎn)到任意函數(shù)指針的功能。

現(xiàn)在,蘋果公司已經(jīng)開(kāi)源了Objective-C 的運(yùn)行時(shí)代碼。你可以在蘋果公司的開(kāi)源網(wǎng)站,找到 objc_msgSend的源碼。

圖3 objc_msgSend 全架構(gòu)實(shí)現(xiàn)源代碼文件列表

上圖列出的是所有架構(gòu)的實(shí)現(xiàn),包括 x86_64 等。objc_msgSend 是 iOS 方式執(zhí)行最核心的部分,編程領(lǐng)域的寶藏,值得你深入探究和細(xì)細(xì)品味。

objc_msgSend方法執(zhí)行的邏輯是:先獲取對(duì)象對(duì)應(yīng)類的信息,再獲取方法的緩存,根據(jù)方法的selector 查找函數(shù)指針,經(jīng)過(guò)異常錯(cuò)誤處理后,最后跳到對(duì)應(yīng)函數(shù)的實(shí)現(xiàn)。

按照這個(gè)邏輯去看源碼會(huì)更加清晰,更容易注意到實(shí)現(xiàn)細(xì)節(jié)。閱讀 objc_msgSend 源碼是編寫(xiě)方法級(jí)耗時(shí)工具的一個(gè)必要的環(huán)節(jié),后面還需要編寫(xiě)一些對(duì)應(yīng)的匯編代碼。

接下來(lái),我們?cè)倏纯丛趺?hook objc_msgSend 方法?

Facebook 開(kāi)源了一個(gè)庫(kù),可以在iOS上運(yùn)行的Mach-O二進(jìn)制文件中動(dòng)態(tài)地重新綁定符號(hào),這個(gè)庫(kù)叫 fishhook。你可以在GitHub 上,查看fishhook的代碼。

fishhook 實(shí)現(xiàn)的大致思路是,通過(guò)重新綁定符號(hào),可以實(shí)現(xiàn)對(duì) c 方法的 hook。dyld 是通過(guò)更新 Mach-O 二進(jìn)制的 __DATA segment 特定的部分中的指針來(lái)綁定 lazy 和 non-lazy 符號(hào),通過(guò)確認(rèn)傳遞給 rebind_symbol 里每個(gè)符號(hào)名稱更新的位置,就可以找出對(duì)應(yīng)替換來(lái)重新綁定這些符號(hào)。

下面,我針對(duì) fishhook 里的關(guān)鍵代碼,和你具體說(shuō)下 fishhook 的實(shí)現(xiàn)原理。

首先,遍歷 dyld 里的所有image,取出 image header 和 slide。代碼如下:

if (!_rebindings_head-\u0026gt;next) { _dyld_register_func_for_add_image(_rebind_symbols_for_image);} else { uint32_t c = _dyld_image_count(); // 遍歷所有 image for (uint32_t i = 0; i \u0026lt; c; i++) { // 讀取 image header 和 slider _rebind_symbols_for_image(_dyld_get_image_header(i), _dyld_get_image_vmaddr_slide(i)); }}

接下來(lái),找到符號(hào)表相關(guān)的 command,包括 linkedit segment command、symtab command 和 dysymtab command。代碼如下:

segment_command_t *cur_seg_cmd;segment_command_t *linkedit_segment = NULL;struct symtab_command* symtab_cmd = NULL;struct dysymtab_command* dysymtab_cmd = NULL;uintptr_t cur = (uintptr_t)header + sizeof(mach_header_t);for (uint i = 0; i \u0026lt; header-\u0026gt;ncmds; i++, cur += cur_seg_cmd-\u0026gt;cmdsize) { cur_seg_cmd = (segment_command_t *)cur; if (cur_seg_cmd-\u0026gt;cmd == LC_SEGMENT_ARCH_DEPENDENT) { if (strcmp(cur_seg_cmd-\u0026gt;segname, SEG_LINKEDIT) == 0) { // linkedit segment command linkedit_segment = cur_seg_cmd; } } else if (cur_seg_cmd-\u0026gt;cmd == LC_SYMTAB) { // symtab command symtab_cmd = (struct symtab_command*)cur_seg_cmd; } else if (cur_seg_cmd-\u0026gt;cmd == LC_DYSYMTAB) { // dysymtab command dysymtab_cmd = (struct dysymtab_command*)cur_seg_cmd; }}

然后,獲得 base 和 indirect 符號(hào)表。實(shí)現(xiàn)代碼如下:

// 找到 base 符號(hào)表的地址uintptr_t linkedit_base = (uintptr_t)slide + linkedit_segment-\u0026gt;vmaddr - linkedit_segment-\u0026gt;fileoff;nlist_t *symtab = (nlist_t *)(linkedit_base + symtab_cmd-\u0026gt;symoff);char *strtab = (char *)(linkedit_base + symtab_cmd-\u0026gt;stroff);// 找到 indirect 符號(hào)表uint32_t *indirect_symtab = (uint32_t *)(linkedit_base + dysymtab_cmd-\u0026gt;indirectsymoff);

最后,有了符號(hào)表和傳入的方法替換數(shù)組,就可以進(jìn)行符號(hào)表訪問(wèn)指針地址的替換了,具體實(shí)現(xiàn)如下:

uint32_t *indirect_symbol_indices = indirect_symtab + section-\u0026gt;reserved1;void **indirect_symbol_bindings = (void **)((uintptr_t)slide + section-\u0026gt;addr);for (uint i = 0; i \u0026lt; section-\u0026gt;size / sizeof(void *); i++) { uint32_t symtab_index = indirect_symbol_indices[i]; if (symtab_index == INDIRECT_SYMBOL_ABS || symtab_index == INDIRECT_SYMBOL_LOCAL || symtab_index == (INDIRECT_SYMBOL_LOCAL | INDIRECT_SYMBOL_ABS)) { continue; } uint32_t strtab_offset = symtab[symtab_index].n_un.n_strx; char *symbol_name = strtab + strtab_offset; if (strnlen(symbol_name, 2) \u0026lt; 2) { continue; } struct rebindings_entry *cur = rebindings; while (cur) { for (uint j = 0; j \u0026lt; cur-\u0026gt;rebindings_nel; j++) { if (strcmp(\u0026amp;symbol_name[1], cur-\u0026gt;rebindings[j].name) == 0) { if (cur-\u0026gt;rebindings[j].replaced != NULL \u0026amp;\u0026amp; indirect_symbol_bindings[i] != cur-\u0026gt;rebindings[j].replacement) { *(cur-\u0026gt;rebindings[j].replaced) = indirect_symbol_bindings[i]; } // 符號(hào)表訪問(wèn)指針地址的替換 indirect_symbol_bindings[i] = cur-\u0026gt;rebindings[j].replacement; goto symbol_loop; } } cur = cur-\u0026gt;next; }symbol_loop:;

以上,就是 fishhook 的實(shí)現(xiàn)原理了。這里的每一步,都對(duì)應(yīng)有代碼實(shí)現(xiàn),你可以點(diǎn)擊文稿查看相應(yīng)的代碼。fishhook 是對(duì)底層的操作,其中查找符號(hào)表的過(guò)程和堆棧符號(hào)化實(shí)現(xiàn)原理基本類似,了解了其中原理對(duì)于理解可執(zhí)行文件 Mach-O 內(nèi)部結(jié)構(gòu)會(huì)有很大的幫助。

接下來(lái),我們?cè)倏匆粋€(gè)問(wèn)題:只靠 fishhook 就能夠搞定 objc_msgSend 的 hook 了嗎?

當(dāng)然還不夠。我前面也說(shuō)了,objc_msgSend 是用匯編語(yǔ)言實(shí)現(xiàn)的,所以我們還需要從匯編層面多加點(diǎn)料。

你需要先實(shí)現(xiàn)兩個(gè)方法 pushCallRecord 和 popCallRecord,來(lái)分別記錄 objc_msgSend 方法調(diào)用前后的時(shí)間,然后相減就能夠得到方法的執(zhí)行耗時(shí)。

下面我針對(duì)arm64架構(gòu),編寫(xiě)一個(gè)可保留未知參數(shù)并跳轉(zhuǎn)到 c 中任意函數(shù)指針的匯編代碼,實(shí)現(xiàn)對(duì) objc_msgSend 的 Hook。

arm64 有31個(gè)64 bit 的整數(shù)型寄存器,分別用 x0 到 x30 表示。主要的實(shí)現(xiàn)思路是:

  • 第一,入棧參數(shù),參數(shù)寄存器是 x0~ x7。對(duì)于objc_msgSend方法來(lái)說(shuō),x0 第一個(gè)參數(shù)是傳入對(duì)象,x1 第二個(gè)參數(shù)是選擇器 _cmd。syscall 的 number 會(huì)放到 x8 里。
  • 第二,交換寄存器中保存的參數(shù),將用于返回的寄存器 lr 中的數(shù)據(jù)移到 x1 里。
  • 第三,使用 bl label 語(yǔ)法調(diào)用 pushCallRecord 函數(shù)。
  • 第四,執(zhí)行原始的 objc_msgSend,保存返回值。
  • 第五,使用 bl label 語(yǔ)法調(diào)用 popCallRecord 函數(shù)。

具體的匯編代碼,你可以點(diǎn)擊文稿查看如下所示:

static void replacementObjc_msgSend() { __asm__ volatile ( // sp 是堆棧寄存器,存放棧的偏移地址,每次都指向棧頂。 // 保存 {q0-q7} 偏移地址到 sp 寄存器 \u0026quot;stp q6, q7, [sp, #-32]!\\u0026quot; \u0026quot;stp q4, q5, [sp, #-32]!\\u0026quot; \u0026quot;stp q2, q3, [sp, #-32]!\\u0026quot; \u0026quot;stp q0, q1, [sp, #-32]!\\u0026quot; // 保存 {x0-x8, lr} \u0026quot;stp x8, lr, [sp, #-16]!\\u0026quot; \u0026quot;stp x6, x7, [sp, #-16]!\\u0026quot; \u0026quot;stp x4, x5, [sp, #-16]!\\u0026quot; \u0026quot;stp x2, x3, [sp, #-16]!\\u0026quot; \u0026quot;stp x0, x1, [sp, #-16]!\\u0026quot; // 交換參數(shù). \u0026quot;mov x2, x1\\u0026quot; \u0026quot;mov x1, lr\\u0026quot; \u0026quot;mov x3, sp\\u0026quot; // 調(diào)用 preObjc_msgSend,使用 bl label 語(yǔ)法。bl 執(zhí)行一個(gè)分支鏈接操作,label 是無(wú)條件分支的,是和本指令的地址偏移,范圍是 -128MB 到 +128MB \u0026quot;bl __Z15preObjc_msgSendP11objc_objectmP13objc_selectorP9RegState_\\u0026quot; \u0026quot;mov x9, x0\\u0026quot; \u0026quot;mov x10, x1\\u0026quot; \u0026quot;tst x10, x10\\u0026quot; // 讀取 {x0-x8, lr} 從保存到 sp 棧頂?shù)钠频刂纷x起 \u0026quot;ldp x0, x1, [sp], #16\\u0026quot; \u0026quot;ldp x2, x3, [sp], #16\\u0026quot; \u0026quot;ldp x4, x5, [sp], #16\\u0026quot; \u0026quot;ldp x6, x7, [sp], #16\\u0026quot; \u0026quot;ldp x8, lr, [sp], #16\\u0026quot; // 讀取 {q0-q7} \u0026quot;ldp q0, q1, [sp], #32\\u0026quot; \u0026quot;ldp q2, q3, [sp], #32\\u0026quot; \u0026quot;ldp q4, q5, [sp], #32\\u0026quot; \u0026quot;ldp q6, q7, [sp], #32\\u0026quot; \u0026quot;b.eq Lpassthrough\\u0026quot; // 調(diào)用原始 objc_msgSend。使用 blr xn 語(yǔ)法。blr 除了從指定寄存器讀取新的 PC 值外效果和 bl 一樣。xn 是通用寄存器的64位名稱分支地址,范圍是0到31 \u0026quot;blr x9\\u0026quot; // 保存 {x0-x9} \u0026quot;stp x0, x1, [sp, #-16]!\\u0026quot; \u0026quot;stp x2, x3, [sp, #-16]!\\u0026quot; \u0026quot;stp x4, x5, [sp, #-16]!\\u0026quot; \u0026quot;stp x6, x7, [sp, #-16]!\\u0026quot; \u0026quot;stp x8, x9, [sp, #-16]!\\u0026quot; // 保存 {q0-q7} \u0026quot;stp q0, q1, [sp, #-32]!\\u0026quot; \u0026quot;stp q2, q3, [sp, #-32]!\\u0026quot; \u0026quot;stp q4, q5, [sp, #-32]!\\u0026quot; \u0026quot;stp q6, q7, [sp, #-32]!\\u0026quot; // 調(diào)用 postObjc_msgSend hook. \u0026quot;bl __Z16postObjc_msgSendv\\u0026quot; \u0026quot;mov lr, x0\\u0026quot; // 讀取 {q0-q7} \u0026quot;ldp q6, q7, [sp], #32\\u0026quot; \u0026quot;ldp q4, q5, [sp], #32\\u0026quot; \u0026quot;ldp q2, q3, [sp], #32\\u0026quot; \u0026quot;ldp q0, q1, [sp], #32\\u0026quot; // 讀取 {x0-x9} \u0026quot;ldp x8, x9, [sp], #16\\u0026quot; \u0026quot;ldp x6, x7, [sp], #16\\u0026quot; \u0026quot;ldp x4, x5, [sp], #16\\u0026quot; \u0026quot;ldp x2, x3, [sp], #16\\u0026quot; \u0026quot;ldp x0, x1, [sp], #16\\u0026quot; \u0026quot;ret\\u0026quot; \u0026quot;Lpassthrough:\\u0026quot; // br 無(wú)條件分支到寄存器中的地址 \u0026quot;br x9\u0026quot; );}

現(xiàn)在,你就可以得到每個(gè) Objective-C 方法的耗時(shí)了。接下來(lái),我們?cè)倏纯?strong>怎樣才能夠做到像下圖那樣記錄和展示方法調(diào)用的層級(jí)關(guān)系和順序呢?

圖4 方法調(diào)用層級(jí)和順序

不要著急,我來(lái)一步一步地跟你說(shuō)。

第一步,設(shè)計(jì)兩個(gè)結(jié)構(gòu)體:CallRecord 記錄調(diào)用方法詳細(xì)信息,包括 obj 和 SEL 等;ThreadCallStack 里面,需要用 index 記錄當(dāng)前調(diào)用方法樹(shù)的深度。

有了 SEL 再通過(guò) NSStringFromSelector 就能夠取得方法名,有了 obj 通過(guò) object_getClass 能夠得到 Class ,再用 NSStringFromClass 就能夠獲得類名。結(jié)構(gòu)的完整代碼如下:

// Shared structures.typedef struct CallRecord_ { id obj; //通過(guò) object_getClass 能夠得到 Class 再通過(guò) NSStringFromClass 能夠得到類名 SEL _cmd; //通過(guò) NSStringFromSelector 方法能夠得到方法名 uintptr_t lr; int prevHitIndex; char isWatchHit;} CallRecord;typedef struct ThreadCallStack_ { FILE *file; char *spacesStr; CallRecord *stack; int allocatedLength; int index; //index 記錄當(dāng)前調(diào)用方法樹(shù)的深度 int numWatchHits; int lastPrintedIndex; int lastHitIndex; char isLoggingEnabled; char isCompleteLoggingEnabled;} ThreadCallStack;

第二步,pthread_setspecific() 可以將私有數(shù)據(jù)設(shè)置在指定線程上,pthread_getspecific() 用來(lái)讀取這個(gè)私有數(shù)據(jù)。利用這個(gè)特性,我們就可以將 ThreadCallStack 的數(shù)據(jù)和該線程綁定在一起,隨時(shí)進(jìn)行數(shù)據(jù)存取。代碼如下:

static inline ThreadCallStack * getThreadCallStack() { ThreadCallStack *cs = (ThreadCallStack *)pthread_getspecific(threadKey); //讀取 if (cs == NULL) { cs = (ThreadCallStack *)malloc(sizeof(ThreadCallStack));#ifdef MAIN_THREAD_ONLY cs-\u0026gt;file = (pthread_main_np()) ? newFileForThread() : NULL;#else cs-\u0026gt;file = newFileForThread();#endif cs-\u0026gt;isLoggingEnabled = (cs-\u0026gt;file != NULL); cs-\u0026gt;isCompleteLoggingEnabled = 0; cs-\u0026gt;spacesStr = (char *)malloc(DEFAULT_CALLSTACK_DEPTH + 1); memset(cs-\u0026gt;spacesStr, ' ', DEFAULT_CALLSTACK_DEPTH); cs-\u0026gt;spacesStr[DEFAULT_CALLSTACK_DEPTH] = '\\0'; cs-\u0026gt;stack = (CallRecord *)calloc(DEFAULT_CALLSTACK_DEPTH, sizeof(CallRecord)); //分配 CallRecord 默認(rèn)空間 cs-\u0026gt;allocatedLength = DEFAULT_CALLSTACK_DEPTH; cs-\u0026gt;index = cs-\u0026gt;lastPrintedIndex = cs-\u0026gt;lastHitIndex = -1; cs-\u0026gt;numWatchHits = 0; pthread_setspecific(threadKey, cs); //保存數(shù)據(jù) } return cs;}

第三步,因?yàn)橐涗浬疃?#xff0c;而一個(gè)方法的調(diào)用里會(huì)有更多的方法調(diào)用,所以我們可以在方法的調(diào)用里增加兩個(gè)方法 pushCallRecord 和 popCallRecord,分別記錄方法調(diào)用的開(kāi)始時(shí)間和結(jié)束時(shí)間,這樣才能夠在開(kāi)始時(shí)對(duì)深度加一、在結(jié)束時(shí)減一。

//開(kāi)始時(shí)static inline void pushCallRecord(id obj, uintptr_t lr, SEL _cmd, ThreadCallStack *cs) { int nextIndex = (++cs-\u0026gt;index); //增加深度 if (nextIndex \u0026gt;= cs-\u0026gt;allocatedLength) { cs-\u0026gt;allocatedLength += CALLSTACK_DEPTH_INCREMENT; cs-\u0026gt;stack = (CallRecord *)realloc(cs-\u0026gt;stack, cs-\u0026gt;allocatedLength * sizeof(CallRecord)); cs-\u0026gt;spacesStr = (char *)realloc(cs-\u0026gt;spacesStr, cs-\u0026gt;allocatedLength + 1); memset(cs-\u0026gt;spacesStr, ' ', cs-\u0026gt;allocatedLength); cs-\u0026gt;spacesStr[cs-\u0026gt;allocatedLength] = '\\0'; } CallRecord *newRecord = \u0026amp;cs-\u0026gt;stack[nextIndex]; newRecord-\u0026gt;obj = obj; newRecord-\u0026gt;_cmd = _cmd; newRecord-\u0026gt;lr = lr; newRecord-\u0026gt;isWatchHit = 0;}//結(jié)束時(shí)static inline CallRecord * popCallRecord(ThreadCallStack *cs) { return \u0026amp;cs-\u0026gt;stack[cs-\u0026gt;index--]; //減少深度}

耗時(shí)檢查的完整代碼,你可以在我的開(kāi)源項(xiàng)目里查看,你可以點(diǎn)擊文稿中的鏈接找到這個(gè)項(xiàng)目。在需要檢測(cè)耗時(shí)時(shí)間的地方調(diào)用 [SMCallTrace start],結(jié)束時(shí)調(diào)用 stop 和 save 就可以打印出方法的調(diào)用層級(jí)和耗時(shí)了。你還可以設(shè)置最大深度和最小耗時(shí)檢測(cè),來(lái)過(guò)濾不需要看到的信息。

有了這樣一個(gè)檢查方法耗時(shí)的工具,你就可以在每個(gè)版本開(kāi)發(fā)結(jié)束后執(zhí)行一次檢查,統(tǒng)計(jì)總耗時(shí)以及啟動(dòng)階段每個(gè)方法的耗時(shí),有針對(duì)性地觀察啟動(dòng)速度慢的問(wèn)題。如果你在線上做個(gè)灰度開(kāi)關(guān),還可以監(jiān)控線上啟動(dòng)慢的一些特殊情況。

接下來(lái),我們小結(jié)一下今天的主要內(nèi)容

啟動(dòng)速度優(yōu)化和監(jiān)控的重要性不言而喻,加快 App 的啟動(dòng)速度對(duì)用戶的體驗(yàn)提升是最大的。

啟動(dòng)速度的優(yōu)化也有粗有細(xì):粗上來(lái)講,這需要對(duì)啟動(dòng)階段功能進(jìn)行分類整理,合理地將和首屏無(wú)關(guān)的功能滯后,放到首屏渲染完成之后,保證大頭兒沒(méi)有問(wèn)題;細(xì)的來(lái)講,這就需要些匠人精神,使用合適的工具,針對(duì)每個(gè)方法進(jìn)行逐個(gè)分析、優(yōu)化,每個(gè)階段都做到極致。

總結(jié)

以上是生活随笔為你收集整理的APP启动速度是门面,如何做到极致优化?的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

色全色在线资源网 | 久99久精品 | 色综合久久久久久久久五月 | 最新日韩视频在线观看 | 色婷婷综合成人av | 中文字幕av全部资源www中文字幕在线观看 | 日韩久久精品一区二区 | 亚洲天堂色婷婷 | 久久精品欧美视频 | 欧美激情视频免费看 | 国产资源免费在线观看 | 五月综合激情 | 欧美电影在线观看 | 久久九九视频 | 欧美精品一区二区免费 | 国产精品久久艹 | 久草在线免费电影 | 国产做aⅴ在线视频播放 | 亚洲国产精品资源 | 国产精品福利在线 | 国产视频一二区 | 国产高清视频网 | 色婷婷综合久色 | 色综合人人 | 精品欧美小视频在线观看 | 久草在线免费电影 | 一区二区三区四区不卡 | 丁香五月缴情综合网 | 激情综合亚洲 | 日韩视频免费在线观看 | 亚洲九九影院 | 香蕉在线视频播放网站 | 国产日韩欧美综合在线 | 国产一二区在线观看 | 午夜精品久久久久久久久久久久 | 激情久久久久久久久久久久久久久久 | 国产98色在线 | 日韩 | 久久伊人八月婷婷综合激情 | 麻豆免费在线视频 | 国模一区二区三区四区 | 丁香色婷婷 | a级成人毛片 | 日韩电影中文字幕 | 日日干夜夜爱 | 黄色一级免费电影 | 99久久精品久久久久久动态片 | 日韩欧美在线视频一区二区三区 | 中文字幕国内精品 | 九七视频在线观看 | 人人爽人人爽av | 337p日本大胆噜噜噜噜 | 久久精品欧美一区二区三区麻豆 | 香蕉久草 | www免费黄色| 六月丁香激情网 | 免费观看国产成人 | 色国产精品 | 亚洲精品视频二区 | 欧美日韩一区二区视频在线观看 | 人人插人人射 | 欧美国产精品一区二区 | 黄色一级大片在线观看 | 91九色在线 | 视频精品一区二区三区 | 久久av网址 | 日韩在线视频一区二区三区 | 伊人在线视频 | 国产人成免费视频 | 国产99一区| 久久久久99精品国产片 | 一区二区三区在线免费观看 | 永久免费毛片在线观看 | 久99久在线 | a黄色大片 | 久久一区91 | 久久曰视频 | 欧美精品一区二区三区四区在线 | 国产精品美女久久久久久网站 | 亚洲欧洲xxxx | 美女在线观看av | 黄色av观看 | 亚洲综合色av | 91麻豆精品国产91久久久无需广告 | 国产一区网址 | 欧美精品免费在线观看 | 亚洲激情在线播放 | 精品国产乱码一区二区三区在线 | 久久国产精品免费看 | 成人久久久电影 | 日韩欧美一区二区不卡 | 国产一区二区高清不卡 | 色激情五月 | 久久人人爽人人爽人人片av软件 | 精品国产一区二区三区男人吃奶 | 天天舔夜夜操 | 婷婷五天天在线视频 | wwwwwww色| 久久怡红院 | 国产第页 | 一区二区成人国产精品 | 99se视频在线观看 | 丁香婷婷久久 | 在线激情网 | 成人香蕉视频 | 婷婷九月丁香 | 国产精品视频永久免费播放 | 午夜精品福利在线 | 成人午夜免费剧场 | 丁香激情五月婷婷 | 久久伊人91| 久久全国免费视频 | 中文一区在线 | 精品视频亚洲 | 欧美日韩亚洲国产一区 | 国产一区精品在线 | 黄色资源在线 | 九九久久国产 | 亚洲精品视频一 | 99久久99久久精品国产片果冰 | 国产精品久久亚洲 | 黄色影院在线免费观看 | 久久国产精品成人免费浪潮 | 色婷婷久久一区二区 | 99re在线视频观看 | 一级黄色免费 | 黄色在线观看www | 92精品国产成人观看免费 | 亚洲a免费 | 国产午夜在线观看 | 久草在线资源观看 | 久久亚洲婷婷 | 天天射天天干天天爽 | 三级黄色免费片 | 日韩免费一二三区 | 国产黄色看片 | 玖草影院 | 日韩av免费观看网站 | 爱爱av网站| 日韩区视频 | 欧美在线观看视频免费 | 久久精品99国产国产精 | 美国三级黄色大片 | a天堂最新版中文在线地址 久久99久久精品国产 | 国产精品久久久久久久久久久免费 | 亚洲人成人99网站 | 超碰97在线资源站 | 精品欧美一区二区精品久久 | 色婷婷亚洲婷婷 | 国产黄色片免费观看 | 日韩在线视频线视频免费网站 | 日韩成人欧美 | 亚洲视频 在线观看 | 亚洲天堂网在线观看视频 | 欧美日韩不卡一区二区三区 | 久久久精品 | 97碰在线视频 | 亚洲精品国产成人av在线 | 性日韩欧美在线视频 | 最新av在线免费观看 | 婷婷五月在线视频 | avove黑丝| 亚洲精品玖玖玖av在线看 | 亚洲精品影院在线观看 | 日韩网站中文字幕 | 欧美日韩激情视频8区 | 亚洲国产日韩在线 | 人人澡人人爽 | 久久精品视频4 | 亚洲 欧美 日韩 综合 | 在线视频观看成人 | 欧美淫aaa免费观看 日韩激情免费视频 | 亚洲高清资源 | 草久久av | 天天操天天操天天 | 成人免费在线电影 | 青青草在久久免费久久免费 | 午夜在线免费视频 | 亚洲涩综合 | 97香蕉超级碰碰久久免费软件 | 又黄又网站| 久久国产免费 | 日韩免费在线观看视频 | 国产精品久久久久久久久久新婚 | 99r在线观看| 五月婷婷激情六月 | 国产97视频 | 国产精品99久久久久久久久 | 日韩大片在线播放 | 久久躁日日躁aaaaxxxx | 黄色99视频| av色影院| 亚洲视频久久久久 | 国产这里只有精品 | 婷婷色在线观看 | 麻豆视频免费网站 | 日韩免费专区 | 天天激情天天干 | 国产成在线观看免费视频 | 久久成人午夜视频 | 99免费在线观看视频 | 久久久不卡影院 | 午夜精品久久久久久久久久久久 | 免费在线一区二区 | 国产最新在线观看 | 在线视频91 | 免费高清在线一区 | 91在线公开视频 | 久久黄色美女 | av片在线看 | 中文字幕在线观看网 | 欧美analxxxx| 天天久久综合 | 亚洲va欧美 | 狠狠操操| 一区二区三区四区久久 | 久久激情视频 久久 | 日韩成人av在线 | 日韩精品中文字幕av | 久草在线这里只有精品 | 日韩亚洲国产中文字幕 | 五月天免费网站 | 九九视频热 | 一区二区三区免费在线播放 | 国产精品精品 | 久久久综合色 | 亚洲美女精品区人人人人 | 91久久久国产精品 | 亚洲精品成人网 | 欧美日韩综合在线观看 | 国产97在线观看 | 永久免费毛片 | 中文字幕免费成人 | 天天射天天搞 | 国产精品一区二区三区四 | 91av在| 国产麻豆成人传媒免费观看 | 久草免费看 | 国产精品情侣视频 | 99精品国产一区二区三区麻豆 | 精品久久久久_ | 欧美一区二区三区在线播放 | 精品视频在线观看 | 麻豆精品91| 中文字幕在线观看一区 | 亚洲精品大片www | 中文字幕日韩无 | 久久精品看 | 国产精品video | 久久综合影视 | av三级av | 婷婷日韩| 精品国产美女在线 | 国产视频在线免费 | 色综合国产 | 手机av在线不卡 | 91在线播放国产 | 国内精品久久久久久久久久 | 国产黄色一级片在线 | 国产资源网 | 国产精品毛片一区二区在线 | 色婷婷激情电影 | 成人av久久 | 天天射天天爱天天干 | 丝袜制服天堂 | 免费成人黄色av | 中文字幕免费高清 | 精品国产一区二区三区四区vr | 国产视频在线看 | 日韩激情三级 | 婷婷视频在线 | 亚洲精品午夜久久久久久久久久久 | 中文字幕在线免费看 | 久久久久国产精品一区 | 成人久久18免费网站 | www.天天色 | 97国产精品一区二区 | 久久九九精品久久 | 免费在线观看成人小视频 | 国产精品视频观看 | 成人av免费在线看 | 国产精品久久久久久久久久久杏吧 | 四虎影院在线观看av | 成在人线av | 久久99国产精品久久 | 婷婷丁香激情综合 | 婷婷六月在线 | 亚洲免费在线观看视频 | 久久国产欧美日韩 | 欧美性色黄 | 亚洲电影图片小说 | 夜夜夜夜夜夜操 | 爱色婷婷 | 国内揄拍国内精品 | 久久综合射 | 精品国产一区二区三区久久 | 激情影院在线观看 | 96国产在线 | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 久久精品日本啪啪涩涩 | 最新真实国产在线视频 | 国产在线不卡精品 | 亚洲精品婷婷 | 国产精品自拍在线 | 国产精品一区二区av日韩在线 | 狠狠躁天天躁 | 欧美日本日韩aⅴ在线视频 插插插色综合 | 国产精品自在欧美一区 | 又色又爽又黄高潮的免费视频 | 久久国产成人午夜av影院潦草 | japanesexxxxfreehd乱熟| 欧美日韩中文另类 | 久久精品久久久久电影 | 色网站在线 | 久久国产电影院 | 日韩免费三区 | 亚洲视屏在线播放 | 日韩色在线 | 九九交易行官网 | 日韩高清片 | 国产91全国探花系列在线播放 | 在线亚洲欧美视频 | 伊人影院得得 | www日韩在线 | 麻豆视频国产精品 | 日本精品在线视频 | 国产一区二区视频在线 | 精品网站999www| 国产色网站 | 蜜臀久久99精品久久久久久网站 | sesese图片| 在线一级片 | av视屏在线 | av在线电影网站 | 亚州精品天堂中文字幕 | 国产成人精品女人久久久 | 久久人人爽人人爽人人片 | 日韩视频免费播放 | 免费观看的av | 一级电影免费在线观看 | 在线观看av大片 | 狠狠干夜夜操 | 国产无限资源在线观看 | av高清网站在线观看 | 一级欧美日韩 | 美女免费视频网站 | 成人精品一区二区三区中文字幕 | 夜夜操天天操 | 成人av在线观 | 一级精品视频在线观看宜春院 | 亚洲1区 在线 | 久久a v电影 | 久草手机视频 | 亚洲国产97在线精品一区 | 成人三级网站在线观看 | av免费在线观看网站 | 欧美乱大交| a在线观看免费视频 | 中文字幕一区二区在线播放 | 中文字幕在线观看视频一区 | 最新国产中文字幕 | 日韩一级电影在线 | 中文字幕在线看视频 | 狠狠色伊人亚洲综合成人 | 一本一道久久a久久综合蜜桃 | 成人午夜电影在线 | 91看片淫黄大片在线播放 | 在线视频一区观看 | 日韩高清在线一区二区 | 亚洲专区欧美 | 日韩精品首页 | 91视频在线播放视频 | 免费福利在线视频 | 国产精品99在线播放 | 亚洲精品在线观看不卡 | 日韩网站在线观看 | 色夜影院 | 亚洲免费精品一区二区 | 91精品久久久久久久久 | 99热只有精品在线观看 | 亚洲精品在线播放视频 | 精品一区二区三区在线播放 | 国产午夜在线 | 国产成人在线综合 | 亚洲做受高潮欧美裸体 | av片子在线观看 | 国产精品理论片在线播放 | 开心色插 | 五月天久久久久久 | 国产亚洲婷婷免费 | 成人午夜精品久久久久久久3d | 超碰97人人爱 | 色五月色开心色婷婷色丁香 | 国产精品福利在线播放 | 91视频久久| 欧美日韩高清在线 | 国产精品午夜8888 | 亚洲精品1234区 | 久久久久国 | 国产人成一区二区三区影院 | 人人搞人人搞 | 综合天天 | 又色又爽又黄 | 久久精品精品电影网 | 久久久久久高潮国产精品视 | 91网免费看 | 天天操夜操视频 | 天天操天天操天天操天天操天天操天天操 | 国产精品自产拍在线观看中文 | 日韩免费看的电影 | h网站免费在线观看 | 中文字幕在线观看视频一区二区三区 | 中文在线字幕免 | 欧美婷婷色 | 亚洲精品88欧美一区二区 | 狠狠狠狠狠色综合 | 国产精品理论片在线观看 | 久久久91精品国产一区二区精品 | 91视频网址入口 | 一级欧美黄| 激情五月婷婷综合 | 在线观看久久久久久 | 国产亚洲视频在线 | 色综合久久综合中文综合网 | 高清精品在线 | 国产精品网址在线观看 | 午夜 免费 | 爱爱av在线 | 久久久首页 | 欧美aⅴ在线观看 | 亚洲精品视频在线看 | 91精品久久久久久综合乱菊 | 九九热视频在线 | 蜜臀精品久久久久久蜜臀 | 五月婷婷中文字幕 | 久久国产系列 | 国产午夜精品一区二区三区在线观看 | 国产在线欧美日韩 | 视频二区 | 中文字幕你懂的 | 9幺看片 | 在线观看免费成人 | 中文字幕第一页在线 | www.夜色321.com | 欧美日韩不卡一区二区三区 | 久久观看最新视频 | 亚洲一级黄色 | 欧美日韩中文视频 | 婷婷九月激情 | 九九九热精品 | 在线观看岛国片 | 日韩xxxx视频 | 黄色成年| 一级片免费在线 | 国产美女视频免费 | 国产精品久久久久久69 | 视频二区在线视频 | 国产精品免费久久久久久 | 久久夜视频 | 久久国产精品影片 | 一本—道久久a久久精品蜜桃 | 日本99热 | 午夜精品福利一区二区三区蜜桃 | 欧美aa一级片 | 国产激情免费 | 国产成人精品一区二区三区 | 91女人18片女毛片60分钟 | 国产传媒一区在线 | 色国产在线 | 午夜在线观看 | 日本在线视频一区二区三区 | 国产1级毛片 | 日韩在线视频一区 | 亚洲精品国偷拍自产在线观看蜜桃 | 成人aaa毛片 | 黄色91在线| 婷婷伊人网| 中文字幕高清在线 | 欧美日韩裸体免费视频 | 天天干天天拍天天操 | 成人欧美一区二区三区在线观看 | 色av资源网 | 91精品资源 | 国产精品自产拍在线观看桃花 | 国产高清视频 | 国产精品白浆 | 最近中文字幕国语免费高清6 | 高清国产一区 | 91av在线视频播放 | 天天射天天操天天 | 亚洲日本va中文字幕 | 国产精品免费在线视频 | 91成人精品国产刺激国语对白 | 成人毛片a | 在线超碰av | 久久www免费视频 | 波多野结衣视频在线 | 亚洲成人二区 | 日韩黄色av网站 | 久久无码精品一区二区三区 | 亚洲五月激情 | av中文字幕免费在线观看 | 九九九热精品免费视频观看网站 | 一级黄色a视频 | 国产精品久久三 | 精精国产xxxx视频在线播放 | 国产视频高清 | 在线观看电影av | 国产午夜视频在线观看 | 在线观看免费中文字幕 | 丝袜网站在线观看 | 免费看一级黄色 | 91精品久久久久久综合乱菊 | 中文在线免费视频 | 天天干天天射天天操 | 久久爱影视i | 亚洲成人免费在线观看 | 亚洲综合色丁香婷婷六月图片 | 中文字幕视频在线播放 | www.天天色 | 午夜在线免费观看 | 一区二区三区在线视频111 | 亚洲成人软件 | 日韩在线视频国产 | 99日精品 | 日韩视频1 | 久草精品免费 | 在线 国产 日韩 | 2023天天干| 人人爽人人爽人人片av免 | 在线看不卡av| 九九免费在线看完整版 | 国产黄色大片免费看 | 九九久久影视 | 韩国av免费观看 | 国产高清中文字幕 | 夜又临在线观看 | 2024国产精品视频 | 天天插天天射 | 久久精品超碰 | 97av影院 | 黄色网址av| 亚洲一区二区三区四区在线视频 | 麻豆视频免费播放 | 九九在线高清精品视频 | 国产一区网址 | 久久av影视 | 国产久草在线观看 | 五月激情久久 | 91视频最新网址 | www.婷婷色 | av在线免费观看网站 | 久久最新视频 | 日韩理论在线观看 | 中文字幕色婷婷在线视频 | 久久综合婷婷国产二区高清 | 超碰在线人人 | 激情黄色av| 亚洲最大成人网4388xx | 日韩av在线高清 | 91成人久久 | 色停停五月天 | 欧美日韩一级久久久久久免费看 | 一级久久久 | 最近日本韩国中文字幕 | 天天看天天干天天操 | 美国av片在线观看 | 国产原创91 | 高清不卡一区二区在线 | 伊人影院av | 操操操av| 久99久精品 | 波多野结衣电影一区二区三区 | 草久电影| 久久国产电影院 | 视频精品一区二区三区 | 激情电影影院 | 欧美日韩免费一区 | 久久精品一区二区三区国产主播 | 黄色网中文字幕 | 免费a现在观看 | 精品999在线观看 | 精品国内 | 成人h动漫精品一区二 | 免费av免费观看 | 综合久久久久久久 | 婷婷色av| 久久久精品网 | 亚洲在线黄色 | 成人av网站在线播放 | 欧洲亚洲女同hd | 在线观看视频一区二区三区 | 亚洲视频axxx | 人人干狠狠干 | 美女网站视频免费都是黄 | 欧美久久久久久久久中文字幕 | 六月丁香激情综合 | 国内精品久久久久久久影视麻豆 | 91久久丝袜国产露脸动漫 | 日韩免费中文字幕 | 91在线看视频免费 | 亚洲日韩中文字幕在线播放 | 色橹橹欧美在线观看视频高清 | 日韩在线| 中文字幕第一 | 亚洲1区 在线 | 在线观看你懂的网址 | 天天色播| 日本二区三区在线 | 久久久久久久影视 | 成年人网站免费在线观看 | 日韩av不卡在线观看 | 日本视频精品 | 91免费网址| 天天爱天天操天天射 | 免费91麻豆精品国产自产在线观看 | 久久国内免费视频 | 中文字幕二区 | 久久午夜网 | 精品国产一区二区三区久久久蜜月 | 久久精品女人毛片国产 | 中文字幕在线免费看线人 | 精品产品国产在线不卡 | 精品免费一区 | 久久久久麻豆v国产 | 久久在线观看视频 | 狠狠干干 | 日韩r级在线 | 亚洲1级片| 精精国产xxxx视频在线播放 | av中文天堂在线 | 在线 日韩 av | 天天干天天插伊人网 | 一级黄色免费网站 | 日韩一级成人av | 亚洲免费在线观看视频 | 久久1区 | 九色视频网址 | 97在线视频观看 | www.婷婷com| 国产精品一区二区三区在线 | 成人黄大片 | 在线婷婷 | 日韩免费视频线观看 | 国产精品美女久久久久久2018 | 精品久久久久久久 | .国产精品成人自产拍在线观看6 | 久草久热 | 亚洲视频在线观看网站 | 久久激情影院 | 免费观看9x视频网站在线观看 | 日本大片免费观看在线 | 亚洲精品免费在线播放 | 色妞色视频一区二区三区四区 | 国产中文字幕视频在线 | 色综合久久久久综合 | 精品久久国产精品 | 激情五月婷婷综合网 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 久久中文字幕导航 | 天天干干 | www.狠狠干 | 岛国av在线 | 精品美女在线视频 | 九九久久久久久久久激情 | 五月激情丁香图片 | 最近中文字幕视频完整版 | 黄网站色视频 | 国产视频一区二区在线播放 | 色综合久久中文综合久久牛 | av免费观看网址 | 久久久精品欧美一区二区免费 | 一区二区三区在线观看中文字幕 | 精品久久国产 | 天天射天天操天天 | 在线观看911视频 | 成人丁香花| 久久精品这里精品 | 五月视频| 日韩成人在线一区二区 | 久久毛片视频 | 18做爰免费视频网站 | 成人a在线观看 | 97视频中文字幕 | 97在线影院 | 四川妇女搡bbbb搡bbbb搡 | 91精品小视频 | 综合黄色网| av日韩av| 日本99热| 91av久久| 国产麻豆成人传媒免费观看 | 日本中文一区二区 | 丁香五月亚洲综合在线 | 婷婷在线视频观看 | 久草爱视频 | 日本视频不卡 | 中文字幕在线一区观看 | 欧美性生爱 | 91爱爱视频 | 久久看片网 | 91超碰在线播放 | 亚洲免费在线观看视频 | 国产91国语对白在线 | 国产色婷婷 | 97超碰成人在线 | 在线视频一二三 | 亚洲精品va | 久久精品久久久久电影 | 六月丁香婷婷网 | 少妇性bbb搡bbb爽爽爽欧美 | 日本韩国精品一区二区在线观看 | 在线观看视频免费播放 | 国产在线精品一区二区 | 久久在线视频精品 | 91污在线观看 | 国产91精品一区二区绿帽 | 中文字幕第一页在线播放 | 一区二区三区中文字幕在线观看 | 欧美性高跟鞋xxxxhd | 狠狠干狠狠久久 | 91九色porny蝌蚪视频 | 精品麻豆 | 开心综合网 | 日韩乱色精品一区二区 | 西西www444 | 91精品老司机久久一区啪 | 黄色片毛片 | 国产成人久久精品77777综合 | 久草精品网 | 手机成人在线 | 最近日本韩国中文字幕 | 日韩av片无码一区二区不卡电影 | 国产麻豆视频在线观看 | 成人免费观看完整版电影 | 久久天天操 | 91在线公开视频 | 激情 婷婷 | 欧美亚洲另类在线视频 | 日韩电影在线看 | 国产精品免费成人 | 91av视频在线免费观看 | 国产一级片免费观看 | 日韩欧美在线国产 | 国产视频在线观看一区 | 成人a v视频 | 亚洲人成免费网站 | 国产裸体bbb视频 | 国产一区二区在线免费播放 | 一本一道久久a久久精品蜜桃 | 亚洲激情综合网 | 久久综合久色欧美综合狠狠 | 在线成人看片 | 色综合天天干 | 一级成人在线 | 激情婷婷色 | 久久黄页 | 欧美在线视频a | 国产中文伊人 | 4438全国亚洲精品在线观看视频 | 国产成人黄色片 | 久久久久久97三级 | 色在线网 | 在线成人av | 又爽又黄又刺激的视频 | 69精品视频 | 91久草视频 | 欧美在线观看禁18 | 日韩精品久久久久久久电影竹菊 | www.av在线.com | 97电影网站| 国产精品色视频 | 国产精品久久久久久久久久ktv | 91精品国产亚洲 | 久久电影色| 日韩在线观看一区 | 黄a网 | 精品爱爱 | 久久精品高清 | 婷婷伊人五月天 | 91免费网| 99av国产精品欲麻豆 | 日韩av免费大片 | 27xxoo无遮挡动态视频 | 中文网丁香综合网 | 欧美一级性生活视频 | 在线亚洲人成电影网站色www | 国产91精品高清一区二区三区 | 久草爱 | 日韩av免费在线电影 | 国产永久免费观看 | 国产美女在线观看 | 日韩高清免费电影 | 国产黑丝袜在线 | 狠狠干网址 | 蜜臀一区二区三区精品免费视频 | 中文字幕日韩免费视频 | 激情视频一区二区三区 | 四虎永久精品在线 | 国产精品久久久久9999吃药 | 国产色拍拍拍拍在线精品 | av免费看av| 果冻av在线 | 久久无码av一区二区三区电影网 | 91字幕| av成人在线播放 | 九九免费观看全部免费视频 | 亚洲精品乱码久久久久久按摩 | 99麻豆视频 | 国产美女免费观看 | 777久久久 | 成年人免费在线观看 | 国产欧美精品在线观看 | 在线小视频国产 | 亚洲专区中文字幕 | 西西人体4444www高清视频 | 97视频总站 | 人人澡人人舔 | 在线成人性视频 | 女人18毛片a级毛片一区二区 | 久久综合婷婷国产二区高清 | 日韩在线观看不卡 | 91最新国产 | 国产特级毛片 | 韩国av一区二区三区在线观看 | 成人aaa毛片| 日本久久久久久久久久久 | 国产在线a | 黄色的视频 | 日韩精品综合在线 | 色资源二区在线视频 | 欧美少妇的秘密 | 在线中文字幕观看 | 夜夜躁日日躁狠狠躁 | 婷婷开心久久网 | 人人插人人草 | 久草在线视频首页 | av亚洲产国偷v产偷v自拍小说 | 国产精品video爽爽爽爽 | 国产精品成人一区二区三区 | 99精品在线观看 | 国产一区二区三区免费观看视频 | 六月激情网 | 国产无遮挡又黄又爽馒头漫画 | 日韩精品欧美一区 | 亚洲天天 | 欧美精品一区二区在线观看 | 成人免费视频观看 | 久久视屏网 | 97激情影院| www久久精品 | 一级一片免费看 | 在线观看成人网 | 久久国产经典视频 | www久久国产 | 国产精品成人av在线 | 天天色天天射天天操 | 成人午夜影院 | 日韩免费看 | 亚洲三级在线 | 欧美久久久久久久久久久久久 | 亚洲一区精品二人人爽久久 | 丁香久久综合 | 黄色国产高清 | 极品嫩模被强到高潮呻吟91 | av成人动漫 | 天天干夜夜夜操天 | 久久这里只精品 | 欧美精品被 | 一级片免费观看 | 久久成人精品视频 | 99热国产在线观看 | 久久国产精品99久久久久久老狼 | 亚洲激情av| 免费在线观看av网址 | 九九热在线精品视频 | 国产精品丝袜久久久久久久不卡 | 最近免费观看的电影完整版 | 人人干干人人 | 久久久国产精品一区二区中文 | 91麻豆免费视频 | 日本中文一级片 | 最新影院 | 欧美久久久久久久久中文字幕 | 国产成人久久精品一区二区三区 | 国产专区在线 | 尤物97国产精品久久精品国产 | 又黄又爽又无遮挡免费的网站 | 色婷婷六月天 | 狠狠躁夜夜av | 久久久久免费 | 午夜在线看片 | 国产亚洲精品久久久久秋 | 男女啪啪网站 | 久9在线 | 色综合久久久久久久久五月 | 国产高清不卡 | 99久久99久久精品免费 | wwwwww色| 黄a网 | 国产精品一区二区久久精品爱涩 | 青青河边草观看完整版高清 | 91福利影院在线观看 | 亚洲日本欧美在线 | 久久久精品 一区二区三区 国产99视频在线观看 | 精品一区二区在线免费观看 | 国产精品 国产精品 | 日韩一区二区在线免费观看 | 国产99色| 最近的中文字幕大全免费版 | 国产黑丝一区二区 | 亚洲美女视频在线观看 | 久久综合免费视频影院 | 成人精品影视 | 黄色的网站在线 | 欧洲精品视频一区二区 | 国产亚洲一区二区在线观看 | 久久久久免费视频 | 国产日韩精品在线观看 | 欧美一区二视频在线免费观看 | 中文字幕精品一区二区三区电影 | 成年在线观看 | 亚洲精品裸体 | 精品国产免费av | 日本一区二区不卡高清 | 又爽又黄在线观看 | 五月婷婷激情 | 人人爽人人爽人人爽人人爽 | 99久久久国产精品美女 | 色香网| 日韩免费观看av | 国产精品原创在线 | 97小视频 | 免费国产黄线在线观看视频 | 国产香蕉视频在线播放 | 久草在线观| 久久久久久久久久免费 | 91成人精品一区在线播放69 | 久久久久久久久久电影 | 亚洲日本国产 | 久久69av| 国产精品一区二区电影 | 国产录像在线观看 | 中文字幕中文中文字幕 | 亚洲美女视频在线观看 | 黄色免费网 | 国产在线视频导航 | www·22com天天操 | 99视频网站 | 狠狠操狠狠干2017 | 久久久久 免费视频 | 日韩视频免费播放 | 91喷水| 在线看片一区 | 日本一区二区不卡高清 | 久久综合九色99 | 精品国产一区二区三区噜噜噜 | 国产xx在线 | 麻豆久久精品 | 日日操天天操夜夜操 | 黄色免费网 | 久保带人| 久久精品影片 | 五月天中文字幕mv在线 | 日本久久久久久久久久久 | 亚洲精品女人久久久 | 五月综合网 | 国产一区二区免费在线观看 | 国产手机av在线 | 亚洲日本韩国一区二区 | 456免费视频 | 五月婷色 | 中文字幕在线观看不卡 | av理论电影| 欧美九九九| 亚洲精品动漫成人3d无尽在线 | 国产精品美乳一区二区免费 | 在线日韩 | 精品一区二三区 | 四虎成人在线 | 免费在线激情电影 | 久久久久久久看片 | 狠狠狠狠狠狠 | 日日夜操 | 久久久精品免费看 | 成人小视频在线观看免费 | 91av中文字幕 | 亚洲人成免费网站 | 婷婷丁香在线视频 | 免费看搞黄视频网站 | 美女网站黄在线观看 | 久久香蕉影视 | 日日夜夜91 | 天天做天天看 | 久青草视频在线观看 | 综合铜03| 亚洲成人精品 | 国产精品99久久免费黑人 | av韩国在线|