Android单元测试 - 几个重要问题
前言
已經(jīng)一個月沒寫文章了,由于9月份在plan國慶旅行計(jì)劃,國慶前前后后去了14天旅行,所以沒時間寫,哈哈。
言歸正傳,上一篇文章《Android單元測試 - 如何開始?》介紹了幾款單元測試框架、Junit & Mockito基本用法、依賴隔離 & Mock概念,本篇主要解答單元測試中幾個重要問題。
在單元測試交流微信群,很多新進(jìn)來的小伙伴,都會幾個大同小異的問題。我們幾個老鳥們答完一次又一次(厚顏無恥地把自己算上^_^),筆者是有點(diǎn)不耐煩了,后來就等其他同學(xué)回答他們.....其實(shí)大家提的問題,歸根到底就是“依賴問題”,jvm依賴還是android依賴?用到native方法報錯怎么辦?靜態(tài)方法怎么解決?
于是呢,筆者決定專門寫一篇文章,來講解這幾個問題。
- 如何解決Android依賴?
- 隔離Native方法
- 解決內(nèi)部new對象
- 靜態(tài)方法
- RxJava異步轉(zhuǎn)同步
1.如何解決Android依賴?
小白:“Presenter中用到TextUtils,運(yùn)行junit時報'java.lang.RuntimeException: Method isEmpty in android.text.TextUtils not mocked'錯誤... 是不是要用robolectric?”
別急,還未到robolectric出場的時候呢!
由于junit運(yùn)行在jvm上,而jdk沒有android源碼,所以TextUtils這些在android sdk中的類,運(yùn)行junit時就引用不上了。既然jdk沒有,我們就自己加唄!
在test/java目錄下,創(chuàng)建android.text.TextUtils類
關(guān)鍵是要個TextUtils同包名、同類名、同方法名。注意不是在main/java下創(chuàng)建,不然會提示Duplicate class found in the file...。單元測試運(yùn)行妥妥的:
原理很簡單,jvm運(yùn)行時會找android.text.TextUtils類,然后找isEmpty方法執(zhí)行。學(xué)過java反射的同學(xué)都知道,只要知道包名類名,就可以拿到Class,知道該類某方法名,就可以獲取Method并執(zhí)行。jvm也是類似的機(jī)制,只要我們給一個包名類名與android sdk相同的類,寫上方法名&參數(shù)&返回值相同的方法,jvm就能編譯并執(zhí)行。
(提示:android的View之類也能這么搞噢)
2.隔離Native方法
小白:“我用到native方法,junit運(yùn)行失敗,robolectric也不支持加載so文件,怎么辦?”
Model類:
單元測試:
run ModelTest... 報錯java.lang.UnsatisfiedLinkError: com.test.unit.Model.nativeMethod()
上篇文章《Android單元測試 - 如何開始?》講述的“依賴隔離”,這里要用到了!
改進(jìn)單元測試:
再run一下,pass了:
這里稍微講講java查找native方法的過程:
1).Model.java全名是com.test.unit.Model.java;
2).調(diào)用native方法nativeMethod()后, jvm會去找C++層com_test_unit_Model.cpp,再找com_test_unit_Model_nativeMethod()方法,并調(diào)用。
在APP運(yùn)行過程,我們會把cpp編譯成so文件,然后讓APP加載到dalvik虛擬機(jī)。但在單元測試中,沒有加載對應(yīng)的so文件,也沒有編譯cpp呀!大牛們可能會嘗試單元測試時加載so文件,但完全沒有必要,也不符合單元測試的原則。
所以,我們可以直接用Mockito框架mock native方法就行啦。實(shí)際上,不僅僅是native方法需要mock,很多依賴的方法、類都要mock,下面會講到更常用的場景。
(參考《Android JNI原理分析》)
3.解決內(nèi)部new對象
小白:“我在Presenter里new Model,Model依賴比較多,會做sql操作,等等.....Presenter依賴Model返回結(jié)果,導(dǎo)致Presenter沒法單元測試?yán)?求大神指點(diǎn)!”
小白C的例子:Model:
Presenter:
錯誤的單元測試:
還是那句話:依賴隔離。我們隔離Model依賴,即mock Model對象,而不是new Model()。
找找以上PresenterTest的問題吧:PresenterTest完全不知道Model的存在,意思是無法mock Model。那么,我們就想辦法把mock Model傳給Presenter——在Presenter構(gòu)造函數(shù)傳參!
改進(jìn)Presenter:
正確的單元測試:
事情就這么解決了。如果你覺得在Activity直接用默認(rèn)Presenter構(gòu)造函數(shù),在構(gòu)造函數(shù)new Model()比較方便,那就保留默認(rèn)構(gòu)造函數(shù)唄。當(dāng)然使用dagger2就不存在多個構(gòu)造函數(shù)了,都是構(gòu)造傳參。
4.靜態(tài)方法
小白:“大神,我在Presenter用到靜態(tài)方法....”筆者:“行了,知道你要說什么。”
Presenter:
解決方法跟上面【解決內(nèi)部new對象】大同小異,核心思想還是依賴隔離。
1).把sign(...)改成非靜態(tài)方法;
2).把SignatureUtils作為成員變量;
3).構(gòu)造方法傳入SignatureUtils;
4).單元測試時,把mock SignatureUtils傳給Presenter。
改進(jìn)后Presenter:
5.RxJava異步轉(zhuǎn)同步
小白:“大神...”
筆者:“為師掐指一算,料汝會遇此劫難。”
小白:(傳說中從入門到出家?)
單元測試
運(yùn)行RxPresenterTest:
你會發(fā)現(xiàn)沒有輸出"test",為什么呢?
由于testRxJava里面,Obserable.subscribeOn(Schedulers.io())把線程切換到io線程,并且delay了1秒,而testTestRxJava()單元測試早已在當(dāng)前線程跑完了。筆者試過,即使去掉delay(1, TimeUnit.SECONDS),還是不會輸出‘test’。
可以看到筆者把.observeOn(AndroidSchedulers.mainThread())注釋掉了,我們把那句代碼加上,再跑一下testTestRxJava(),會報java.lang.RuntimeException: Method getMainLooper in android.os.Looper not mocked.:
這是由于jdk沒有android.os.Looper這個類及相關(guān)依賴。
解決以上兩個問題,我們只要把Schedulers.io()&AndroidSchedulers.mainThread()切換為Schedulers.immediate()就可以了。RxJava開發(fā)團(tuán)隊(duì)已經(jīng)為大家想好了,提供了RxJavaHooks和RxAndroidPlugins兩個hook操作的類。
新建RxTools:
在RxPresenterTest.setUp()加一句RxTools.asyncToSync();:
再跑一次testTestRxJava():
總算輸出"test",感謝上帝啊!(應(yīng)該打賞下筆者吧^_^)
讀者有沒發(fā)現(xiàn)RxTools.asyncToSync()多加了一句RxJavaHooks.setOnComputationScheduler(schedulerFunc),意思將computation線程切換為immediate線程。筆者發(fā)現(xiàn),僅僅添加RxJavaHooks.setOnIOScheduler(schedulerFunc),對于有delay的Obserable還是未通過,于是順手把computation線程也切換了,于是就可以了。
還有RxJavaHooks.reset()和RxAndroidPlugins.getInstance().reset(),筆者發(fā)現(xiàn),當(dāng)運(yùn)行大量單元測試時,有些會失敗,但單獨(dú)運(yùn)行失敗的單元測試,又通過了。百思不得其解后,添加了那兩句.....可以了!
(關(guān)于RxJavaHooks和RxAndroidPlugins的使用,在很久前的文章 《(MVP+RxJava+Retrofit)解耦+Mockito單元測試 經(jīng)驗(yàn)分享》已經(jīng)提及過)
小結(jié)
筆者:“小白同學(xué),現(xiàn)在你踩過的坑,填好未?”
小白:“方丈,啊不,大神,上面幾個問題是解決了,不過還有其他問題。”
筆者:“不挖坑,怎么填坑呢?以后再給你講講其他單元測試的玄機(jī)。”
小白:“......”
本文詳述了幾個單元測試重要問題的解決方法,讀者不難發(fā)現(xiàn),筆者一直強(qiáng)調(diào) 依賴隔離、依賴隔離、依賴隔離,這個概念在單元測試中相當(dāng)重要。還搞不懂這個概念的同學(xué),看多幾次《Android單元測試 - 如何開始?》(又厚顏無恥地廣告),同時在實(shí)踐中不斷回顧這個理念。
只要解決好這幾個問題,Presenter單元測試就不難了。還有本文未提及的sqlite、SharedPreferences單元測試、在后面的文章會給讀者介紹下。
感謝讀者對筆者一直以來的支持,麻煩點(diǎn)贊&隨手轉(zhuǎn)發(fā),好人一世平安。
關(guān)于作者
我是鍵盤男。在廣州生活,在創(chuàng)業(yè)公司上班,猥瑣文藝碼農(nóng)。喜歡科學(xué)、歷史,玩玩投資,偶爾獨(dú)自旅行。希望成為獨(dú)當(dāng)一面的工程師。
作者:鍵盤男 來源:51CTO
總結(jié)
以上是生活随笔為你收集整理的Android单元测试 - 几个重要问题的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 2017第32周一
- 下一篇: 又到中元节 应用宝教你如何打败各种鬼