這篇文章主要介紹了Retrofit源碼的示例分析,具有一定借鑒價(jià)值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。
目前累計(jì)服務(wù)客戶(hù)千余家,積累了豐富的產(chǎn)品開(kāi)發(fā)及服務(wù)經(jīng)驗(yàn)。以網(wǎng)站設(shè)計(jì)水平和技術(shù)實(shí)力,樹(shù)立企業(yè)形象,為客戶(hù)提供成都做網(wǎng)站、網(wǎng)站設(shè)計(jì)、網(wǎng)站策劃、網(wǎng)頁(yè)設(shè)計(jì)、網(wǎng)絡(luò)營(yíng)銷(xiāo)、VI設(shè)計(jì)、網(wǎng)站改版、漏洞修補(bǔ)等服務(wù)。成都創(chuàng)新互聯(lián)始終以務(wù)實(shí)、誠(chéng)信為根本,不斷創(chuàng)新和提高建站品質(zhì),通過(guò)對(duì)領(lǐng)先技術(shù)的掌握、對(duì)創(chuàng)意設(shè)計(jì)的研究、對(duì)客戶(hù)形象的視覺(jué)傳遞、對(duì)應(yīng)用系統(tǒng)的結(jié)合,為客戶(hù)提供更好的一站式互聯(lián)網(wǎng)解決方案,攜手廣大客戶(hù),共同發(fā)展進(jìn)步。基本使用介紹
介紹源碼前,我們先看下Retrofit的基本使用,大致了解下流程,跟著這個(gè)流程來(lái)分析源碼才不會(huì)亂。
1、初始化Retrofit對(duì)象
Retrofit retrofit = new Retrofit.Builder() //使用自定義的mGsonConverterFactory .addConverterFactory(GsonConverterFactory.create()) .baseUrl("http://apis.baidu.com/txapi/") .build();
2、定義接口
public interface APi { @GET("hello/world") Call<News> getNews(@Query("num") String num,@Query("page")String page); }
3、發(fā)起網(wǎng)絡(luò)請(qǐng)求
mApi = retrofit.create(APi.class); Call<News> news = mApi.getNews("1", "10"); news.enqueue(new Callback<News>() { @Override public void onResponse(Call<News> call, Response<News> response) { } @Override public void onFailure(Call<News> call, Throwable t) { } });
Retrofit本質(zhì)分析
看到上面的整個(gè)流程,不去探究源碼的話(huà)肯定一臉懵逼,我就定義了一個(gè)接口,指定了下返回值,為毛這個(gè)接口就可以直接用了?接口的實(shí)現(xiàn)呢?我隨便寫(xiě)一個(gè)返回值,不指定返回Call行不行?待著這些疑問(wèn),我們大致也可以猜出Retrofit是干什么的了。
猜測(cè):Retrofit主要就是為我們定義的接口創(chuàng)造了一個(gè)實(shí)例,然后這個(gè)實(shí)例調(diào)用接口中的方法將我們定義在注解中的值拼裝成發(fā)起http請(qǐng)求所要的信息,最后利用這些信息產(chǎn)生一個(gè)我們?cè)诮涌诜祷刂抵幸?guī)定的對(duì)象,這個(gè)對(duì)象可以用來(lái)發(fā)起真正的請(qǐng)求。
簡(jiǎn)單的講,Retrofit就是把注解中的東西拼成http請(qǐng)求的對(duì)象,然后由這個(gè)對(duì)象去發(fā)起請(qǐng)求。
驗(yàn)證猜測(cè)
是誰(shuí)實(shí)現(xiàn)了這個(gè)接口
發(fā)起網(wǎng)絡(luò)請(qǐng)求時(shí),有這樣一句:
mApi = retrofit.create(APi.class);
很明顯,接口的實(shí)現(xiàn)應(yīng)該是這個(gè)create干的事,我們跟進(jìn)去看看源碼:
public <T> T create(final Class<T> service) { Utils.validateServiceInterface(service); if (validateEagerly) { eagerlyValidateMethods(service); } return (T) Proxy.newProxyInstance(service.getClassLoader(), new Class<?>[] { service }, new InvocationHandler() { private final Platform platform = Platform.get(); @Override public Object invoke(Object proxy, Method method, @Nullable Object[] args) throws Throwable { // If the method is a method from Object then defer to normal invocation. if (method.getDeclaringClass() == Object.class) { return method.invoke(this, args); } if (platform.isDefaultMethod(method)) { return platform.invokeDefaultMethod(method, service, proxy, args); } ServiceMethod<Object, Object> serviceMethod = (ServiceMethod<Object, Object>) loadServiceMethod(method); OkHttpCall<Object> okHttpCall = new OkHttpCall<>(serviceMethod, args); return serviceMethod.adapt(okHttpCall); } }); }
這里我們不用去看一些細(xì)節(jié),只去關(guān)注我們想知道的,這個(gè)Create方法到底干了什么,看到Proxy.newProxyInstance,哦!這就很明顯了,這里接口之所以能夠直接調(diào)用是使用了動(dòng)態(tài)代理技術(shù),產(chǎn)生了一個(gè)代理對(duì)象。binggo,一個(gè)問(wèn)題解決!!!
什么時(shí)候開(kāi)始將注解中參數(shù)拼裝成http請(qǐng)求的信息的?
動(dòng)態(tài)代理是干什么的?(大家最好單獨(dú)去學(xué)習(xí)下Java的動(dòng)態(tài)代理,有時(shí)候非常有用,資料網(wǎng)上大把大把的)
使用動(dòng)態(tài)代理一般是為了攔截方法,好在個(gè)方法執(zhí)行的時(shí)候,在執(zhí)行之前或之后干些自己的事情。這里回顧下Retrofit的使用,通過(guò)Create方法產(chǎn)生代理對(duì)象后直接就調(diào)用定義的方法了。那么,所有的小動(dòng)作必然是在調(diào)用接口方法的時(shí)候干的。具體就是invoke中的方法
if (method.getDeclaringClass() == Object.class) { return method.invoke(this, args); } if (platform.isDefaultMethod(method)) { return platform.invokeDefaultMethod(method, service, proxy, args); } ServiceMethod<Object, Object> serviceMethod = (ServiceMethod<Object, Object>) loadServiceMethod(method); OkHttpCall<Object> okHttpCall = new OkHttpCall<>(serviceMethod, args); return serviceMethod.adapt(okHttpCall);
前兩個(gè)if判斷主要是為了跳過(guò)object和object中內(nèi)置的一些方法,除了這些,剩下的所有方法必然是我們?cè)诮涌谥卸x的,也就是我們要攔截的。那么真正干事的就是三句:
ServiceMethod<Object, Object> serviceMethod = (ServiceMethod<Object, Object>) loadServiceMethod(method); OkHttpCall<Object> okHttpCall = new OkHttpCall<>(serviceMethod, args); return serviceMethod.adapt(okHttpCall);
其中l(wèi)oadServiceMethod()方法就是用來(lái)拼裝http請(qǐng)求信息的,這個(gè)問(wèn)題解決!
如何產(chǎn)生發(fā)起http請(qǐng)求對(duì)象的?
經(jīng)過(guò)上面的分析, OkHttpCall<Object> okHttpCall = new OkHttpCall<>(serviceMethod, args);這就很明顯就是用來(lái)生產(chǎn)這個(gè)對(duì)象的。
如何將對(duì)象轉(zhuǎn)換成我們?cè)诮涌谥兄付ǖ姆祷刂档模?/p>
這里要說(shuō)明下,之前我們定義接口的時(shí)候是這樣的:
@GET("hello/world") Call<News> getNews(@Query("num") String num,@Query("page")String page);
這里的Call和Okhttp的Call其實(shí)很像,其實(shí)連方法都幾乎一樣,但是Retrofit和Rxjava一起用的時(shí)候又可以這樣定義
@GET("book/search") Observable<Book> getSearchBook(@Query("q") String name, @Query("tag") String tag, @Query("start") int start, @Query("count") int count);
這里指定的返回值不再是Call,而是變成了RxJava的Observable,所以這里肯定有一步轉(zhuǎn)換,能將之前拼裝的信息轉(zhuǎn)換成我們指定的對(duì)象,具體核心就是最后一行的serviceMethod.adapt()干的。
感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“Retrofit源碼的示例分析”這篇文章對(duì)大家有幫助,同時(shí)也希望大家多多支持創(chuàng)新互聯(lián),關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,更多相關(guān)知識(shí)等著你來(lái)學(xué)習(xí)!
新聞標(biāo)題:Retrofit源碼的示例分析-創(chuàng)新互聯(lián)
本文URL:http://vcdvsql.cn/article46/iiehg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供關(guān)鍵詞優(yōu)化、面包屑導(dǎo)航、電子商務(wù)、域名注冊(cè)、網(wǎng)站內(nèi)鏈、App設(shè)計(jì)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶(hù)投稿、用戶(hù)轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話(huà):028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容