Retrofit 2.0 官方文檔

Retrofit 2.0

前言##

來(lái)自移動(dòng)支付公司square公司的作品,開(kāi)源世界top5的最小公司颈娜,首先我自己是一個(gè)忠實(shí)廣場(chǎng)粉自晰,okhttp凝化、picasso、greendao酬荞、okio等等~
據(jù)Square CTO Bob Lee的說(shuō)法搓劫,Square已經(jīng)將超過(guò)60個(gè)項(xiàng)目提交到開(kāi)源社區(qū),貢獻(xiàn)了25萬(wàn)行左右的代碼混巧。

原文:Retrofit 2.0: The biggest update yet on the best HTTP Client Library for Android

因?yàn)槠浜?jiǎn)單與出色的性能枪向,Retrofit 是安卓上最流行的HTTP Client庫(kù)之一。

不過(guò)它的缺點(diǎn)是在Retrofit 1.x中沒(méi)有直接取消正在進(jìn)行中任務(wù)的方法咧党。如果你想做這件事必須手動(dòng)殺死秘蛔,而這并不好實(shí)現(xiàn)。

Square幾年前曾許諾這個(gè)功能將在Retrofit 2.0實(shí)現(xiàn)傍衡,但是幾年過(guò)去了仍然沒(méi)有在這個(gè)問(wèn)題上有所更新深员。

API 聲明

接口函數(shù)的注解和參數(shù)表明如何去處理請(qǐng)求
請(qǐng)求方法

每一個(gè)函數(shù)都必須有提供請(qǐng)求方式和相對(duì)URL的Http注解,Retrofit提供了5種內(nèi)置的注解:GET蛙埂、POST倦畅、PUT、DELETE和HEAD绣的,在注解中指定的資源的相對(duì)URL

@GET("users/list")

也可以在URL中指定查詢參數(shù)

@GET("users/list?sort=desc")

URL處理

請(qǐng)求的URL可以在函數(shù)中使用替換塊和參數(shù)進(jìn)行動(dòng)態(tài)更新叠赐,替換塊是{ and }包圍的字母數(shù)字組成的字符串欲账,相應(yīng)的參數(shù)必須使用相同的字符串被@Path進(jìn)行注釋

@GET("group/{id}/users")
List<User> groupList(@Path("id") int groupId);

也可以添加查詢參數(shù)

@GET("group/{id}/users")
List<User> groupList(@Path("id") int groupId, @Query("sort") String sort);

復(fù)雜的查詢參數(shù)可以使用Map進(jìn)行組合

@GET("group/{id}/users")
List<User> groupList(@Path("id") int groupId, @QueryMap Map<String, String> options);

請(qǐng)求體

可以通過(guò)@Body注解指定一個(gè)對(duì)象作為Http請(qǐng)求的請(qǐng)求體

@POST("users/new")
Call<User> createUser(@Body User user);

該對(duì)象將會(huì)被Retroofit實(shí)例指定的轉(zhuǎn)換器轉(zhuǎn)換,如果沒(méi)有添加轉(zhuǎn)換器芭概,則只有RequestBody可用赛不。(轉(zhuǎn)換器的添加在后面介紹)
FORM ENCODED 和 MULTIPART

函數(shù)也可以聲明為發(fā)送form-encoded和multipart數(shù)據(jù)。
當(dāng)函數(shù)有@FormUrlEncoded注解的時(shí)候罢洲,將會(huì)發(fā)送form-encoded數(shù)據(jù)俄删,每個(gè)鍵-值對(duì)都要被含有名字的@Field注解和提供值的對(duì)象所標(biāo)注

@FormUrlEncoded
@POST("user/edit")
Call<User> updateUser(@Field("first_name") String first, @Field("last_name") String last);

當(dāng)函數(shù)有@Multipart注解的時(shí)候,將會(huì)發(fā)送multipart數(shù)據(jù)奏路,Parts都使用@Part注解進(jìn)行聲明

@Multipart
@PUT("user/photo")
Call<User> updateUser(@Part("photo") RequestBody photo, @Part("description") RequestBody description);

Multipart parts要使用Retrofit的眾多轉(zhuǎn)換器之一或者實(shí)現(xiàn)RequestBody來(lái)處理自己的序列化畴椰。
Header處理

可以使用@Headers注解給函數(shù)設(shè)置靜態(tài)的header

@Headers("Cache-Control: max-age=640000")
@GET("widget/list")
Call<List<Widget>> widgetList();

@Headers({
    "Accept: application/vnd.github.v3.full+json",
    "User-Agent: Retrofit-Sample-App"
})
@GET("users/{username}")
Call<User> getUser(@Path("username") String username);

需要注意的是:header不能被互相覆蓋。所有具有相同名字的header將會(huì)被包含到請(qǐng)求中鸽粉。

可以使用@Header注解動(dòng)態(tài)的更新一個(gè)請(qǐng)求的header斜脂。必須給@Header提供相應(yīng)的參數(shù),如果參數(shù)的值為空header將會(huì)被忽略触机,否則就調(diào)用參數(shù)值的toString()方法并使用返回結(jié)果

@GET("user")
Call<User> getUser(@Header("Authorization") String authorization)

使用OkHttp攔截器可以指定需要的header給每一個(gè)Http請(qǐng)求

OkHttpClient client = new OkHttpClient();
client.networkInterceptors().add(new Interceptor() {
    @Override
    public com.squareup.okhttp.Response intercept(Chain chain) throws IOException {
        com.squareup.okhttp.Response response = chain.proceed(chain.request());
        // Do anything with response here

        return response;
    }
});
Retrofit retrofit = new Retrofit.Builder()
        .baseUrl(BASE_URL)
        ...
        .client(client)
        .build();

新的Service定義方式帚戳,不再有同步和異步之分
關(guān)于在Retrofit 1.9中service 接口的定義,如果你想定義一個(gè)同步的函數(shù)儡首,你應(yīng)該這樣定義:

/* Synchronous in Retrofit 1.9 */
 
public interface APIService {
 
    @POST("/list")
    Repo loadRepo();
 
}

而定義一個(gè)異步的則是這樣:

/* Asynchronous in Retrofit 1.9 */
 
public interface APIService {
 
    @POST("/list")
    void loadRepo(Callback<Repo> cb);
 
}

但是在Retrofit 2.0上片任,只能定義一個(gè)模式,因此要簡(jiǎn)單得多蔬胯。

import retrofit.Call;
 
/* Retrofit 2.0 */
 
public interface APIService {
 
    @POST("/list")
    Call<Repo> loadRepo();
 
}

而創(chuàng)建service 的方法也變得和OkHttp的模式一模一樣对供。如果要調(diào)用同步請(qǐng)求,只需調(diào)用execute氛濒;而發(fā)起一個(gè)異步請(qǐng)求則是調(diào)用enqueue产场。

同步請(qǐng)求

// Synchronous Call in Retrofit 2.0
 
Call<Repo> call = service.loadRepo();
Repo repo = call.execute();

以上的代碼會(huì)阻塞線程,因此你不能在安卓的主線程中調(diào)用舞竿,不然會(huì)面臨NetworkOnMainThreadException京景。如果你想調(diào)用execute方法,請(qǐng)?jiān)诤笈_(tái)線程執(zhí)行骗奖。

異步請(qǐng)求

// Synchronous Call in Retrofit 2.0
 
Call<Repo> call = service.loadRepo();
call.enqueue(new Callback<Repo>() {
    @Override
    public void onResponse(Response<Repo> response) {
        // Get result Repo from response.body()
    }
 
    @Override
    public void onFailure(Throwable t) {
 
    }
});

以上代碼發(fā)起了一個(gè)在后臺(tái)線程的請(qǐng)求并從response 的response.body()方法中獲取一個(gè)結(jié)果對(duì)象确徙。注意這里的onResponse和onFailure方法是在主線程中調(diào)用的。

我建議你使用enqueue执桌,它最符合 Android OS的習(xí)慣鄙皇。

取消正在進(jìn)行中的業(yè)務(wù)
service 的模式變成Call的形式的原因是為了讓正在進(jìn)行的事務(wù)可以被取消。要做到這點(diǎn)鼻吮,你只需調(diào)用call.cancel()育苟。

call.cancel();
事務(wù)將會(huì)在之后立即被取消。好簡(jiǎn)單嘿嘿椎木!

Converter現(xiàn)在從Retrofit中刪除
在Retrofit 1.9中违柏,GsonConverter 包含在了package 中而且自動(dòng)在RestAdapter創(chuàng)建的時(shí)候被初始化。這樣來(lái)自服務(wù)器的son結(jié)果會(huì)自動(dòng)解析成定義好了的Data Access Object(DAO)

但是在Retrofit 2.0中香椎,Converter 不再包含在package 中了漱竖。你需要自己插入一個(gè)Converter 不然的話Retrofit 只能接收字符串結(jié)果。同樣的畜伐,Retrofit 2.0也不再依賴于Gson 馍惹。

如果你想接收json 結(jié)果并解析成DAO,你必須把Gson Converter 作為一個(gè)獨(dú)立的依賴添加進(jìn)來(lái)玛界。

compile 'com.squareup.retrofit:converter-gson:2.0.0-beta1'

然后使用addConverterFactory把它添加進(jìn)來(lái)万矾。注意RestAdapter的別名仍然為Retrofit。

Retrofit retrofit = new Retrofit.Builder()
        .baseUrl("http://api.nuuneoi.com/base/")
        .addConverterFactory(GsonConverterFactory.create())
        .build();
 
service = retrofit.create(APIService.class);

這里是Square提供的官方Converter modules列表慎框。選擇一個(gè)最滿足你需求的良狈。

Gson: com.squareup.retrofit:converter-gson
Jackson: com.squareup.retrofit:converter-jackson
Moshi: com.squareup.retrofit:converter-moshi
Protobuf: com.squareup.retrofit:converter-protobuf
Wire: com.squareup.retrofit:converter-wire
Simple XML: com.squareup.retrofit:converter-simplexml

你也可以通過(guò)實(shí)現(xiàn)Converter.Factory接口來(lái)創(chuàng)建一個(gè)自定義的converter 。

我比較贊同這種新的模式笨枯。它讓Retrofit對(duì)自己要做的事情看起來(lái)更清晰薪丁。

自定義Gson對(duì)象
為了以防你需要調(diào)整json里面的一些格式,比如馅精,Date Format严嗜。你可以創(chuàng)建一個(gè)Gson 對(duì)象并把它傳遞給GsonConverterFactory.create()稼稿。

Gson gson = new GsonBuilder()
        .setDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")
        .create();
 
Retrofit retrofit = new Retrofit.Builder()
        .baseUrl("http://api.nuuneoi.com/base/")
        .addConverterFactory(GsonConverterFactory.create(gson))
        .build();
 
service = retrofit.create(APIService.class);

完成扮叨。

新的URL定義方式
Retrofit 2.0使用了新的URL定義方式。Base URL與@Url 不是簡(jiǎn)單的組合在一起而是和"<a href="...">"的處理方式一致瞳遍。用下面的幾個(gè)例子闡明压彭。

1称近、ps:貌似第二個(gè)才符合習(xí)慣。

對(duì)于 Retrofit 2.0中新的URL定義方式哮塞,這里是我的建議:

  • Base URL: 總是以 /結(jié)尾

  • @Url: 不要以 / 開(kāi)頭

比如

public interface APIService {
 
    @POST("user/list")
    Call<Users> loadUsers();
 
}
 
public void doSomething() {
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl("http://api.nuuneoi.com/base/")
            .addConverterFactory(GsonConverterFactory.create())
            .build();
 
    APIService service = retrofit.create(APIService.class);
}

以上代碼中的loadUsers會(huì)從 http://api.nuuneoi.com/base/user/list獲取數(shù)據(jù)刨秆。

而且在Retrofit 2.0中我們還可以在@Url里面定義完整的URL:

public interface APIService {
 
    @POST("http://api.nuuneoi.com/special/user/list")
    Call<Users> loadSpecialUsers();
 
}

這種情況下Base URL會(huì)被忽略。

可以看到在URL的處理方式上發(fā)生了很大變化忆畅。它和前面的版本完全不同衡未。如果你想把代碼遷移到Retrofit 2.0,別忘了修正URL部分的代碼家凯。

現(xiàn)在需要OkHttp的支持
OkHttp 在Retrofit 1.9里是可選的缓醋。如果你想讓Retrofit 使用OkHttp 作為HTTP 連接接口,你需要手動(dòng)包含okhttp 依賴绊诲。

但是在Retrofit 2.0中送粱,OkHttp 是必須的,并且自動(dòng)設(shè)置為了依賴掂之。下面的代碼是從Retrofit 2.0的pom文件中抓取的抗俄。你不需要再做任何事情了脆丁。

<dependencies>
  <dependency>
    <groupId>com.squareup.okhttp</groupId>
    <artifactId>okhttp</artifactId>
  </dependency>
 
  ...
</dependencies>

為了讓OkHttp 的Call模式成為可能,在Retrofit 2.0中OkHttp 自動(dòng)被用作HTTP 接口动雹。

即使response存在問(wèn)題onResponse依然被調(diào)用
在Retrofit 1.9中槽卫,如果獲取的 response 不能背解析成定義好的對(duì)象,則會(huì)調(diào)用failure胰蝠。但是在Retrofit 2.0中歼培,不管 response 是否能被解析。onResponse總是會(huì)被調(diào)用茸塞。但是在結(jié)果不能被解析的情況下躲庄,response.body()會(huì)返回null。別忘了處理這種情況钾虐。

如果response存在什么問(wèn)題噪窘,比如404什么的,onResponse也會(huì)被調(diào)用禾唁。你可以從response.errorBody().string()中獲取錯(cuò)誤信息的主體效览。

Response/Failure 邏輯和Retrofit 1.9差別很大。如果你決定遷移到Retrofit 2.0荡短,注意小心謹(jǐn)慎的處理這些情況丐枉。

缺少INTERNET權(quán)限會(huì)導(dǎo)致SecurityException異常
在Retrofit 1.9中,如果你忘記在AndroidManifest.xml文件中添加INTERNET權(quán)限掘托。異步請(qǐng)求會(huì)直接進(jìn)入failure回調(diào)方法瘦锹,得到PERMISSION DENIED 錯(cuò)誤消息。沒(méi)有任何異常被拋出闪盔。

但是在Retrofit 2.0中弯院,當(dāng)你調(diào)用call.enqueue或者call.execute,將立即拋出SecurityException泪掀,如果你不使用try-catch會(huì)導(dǎo)致崩潰听绳。

這類似于在手動(dòng)調(diào)用HttpURLConnection時(shí)候的行為。不過(guò)這不是什么大問(wèn)題异赫,因?yàn)楫?dāng)INTERNET權(quán)限添加到了 AndroidManifest.xml中就沒(méi)有什么需要考慮的了椅挣。

Use an Interceptor from OkHttp
在Retrofit 1.9中,你可以使用RequestInterceptor來(lái)攔截一個(gè)請(qǐng)求塔拳,但是它已經(jīng)從Retrofit 2.0 移除了鼠证,因?yàn)镠TTP連接層已經(jīng)轉(zhuǎn)為OkHttp。

結(jié)果就是靠抑,現(xiàn)在我們必須轉(zhuǎn)而使用OkHttp里面的Interceptor量九。首先你需要使用Interceptor創(chuàng)建一個(gè)OkHttpClient對(duì)象,如下:

OkHttpClient client = new OkHttpClient();
client.interceptors().add(new Interceptor() {
    @Override
    public Response intercept(Chain chain) throws IOException {
        Response response = chain.proceed(chain.request());
 
        // Do anything with response here
 
        return response;
    }
});

然后傳遞創(chuàng)建的client到Retrofit的Builder鏈中。

Retrofit retrofit = new Retrofit.Builder()
        .baseUrl("http://api.nuuneoi.com/base/")
        .addConverterFactory(GsonConverterFactory.create())
        .client(client)
        .build();

以上為全部?jī)?nèi)容荠列。

學(xué)習(xí)關(guān)于OkHttp Interceptor的知識(shí)类浪,請(qǐng)到OkHttp Interceptors。

RxJava Integration with CallAdapter
除了使用Call模式來(lái)定義接口弯予,我們也可以定義自己的type戚宦,比如MyCall个曙。锈嫩。我們把Retrofit 2.0的這個(gè)機(jī)制稱為CallAdapter。

Retrofit團(tuán)隊(duì)有已經(jīng)準(zhǔn)備好了的CallAdapter module垦搬。其中最著名的module可能是為RxJava準(zhǔn)備的CallAdapter呼寸,它將作為Observable返回。要使用它猴贰,你的項(xiàng)目依賴中必須包含兩個(gè)modules对雪。

compile 'com.squareup.retrofit:adapter-rxjava:2.0.0-beta1'
compile 'io.reactivex:rxandroid:1.0.1'

Sync Gradle并在Retrofit Builder鏈表中如下調(diào)用addCallAdapterFactory:

Retrofit retrofit = new Retrofit.Builder()
        .baseUrl("http://api.nuuneoi.com/base/")
        .addConverterFactory(GsonConverterFactory.create())
        .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
        .build();

你的Service接口現(xiàn)在可以作為Observable返回了!

Retrofit retrofit = new Retrofit.Builder()
        .baseUrl("http://api.nuuneoi.com/base/")
        .addConverterFactory(GsonConverterFactory.create())
        .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
        .build();

你可以完全像RxJava那樣使用它米绕,如果你想讓subscribe部分的代碼在主線程被調(diào)用瑟捣,需要把observeOn(AndroidSchedulers.mainThread())添加到鏈表中。

Observable<DessertItemCollectionDao> observable = service.loadDessertListRx();
 
observable.observeOn(AndroidSchedulers.mainThread())
    .subscribe(new Subscriber<DessertItemCollectionDao>() {
        @Override
        public void onCompleted() {
            Toast.makeText(getApplicationContext(),
                    "Completed",
                    Toast.LENGTH_SHORT)
                .show();
        }
 
        @Override
        public void onError(Throwable e) {
            Toast.makeText(getApplicationContext(),
                    e.getMessage(),
                    Toast.LENGTH_SHORT)
                .show();
        }
 
        @Override
        public void onNext(DessertItemCollectionDao dessertItemCollectionDao) {
            Toast.makeText(getApplicationContext(),
                    dessertItemCollectionDao.getData().get(0).getName(),
                    Toast.LENGTH_SHORT)
                .show();
        }
    });

完成栅干!我相信RxJava的粉絲對(duì)這個(gè)變化相當(dāng)滿意迈套。

總結(jié)
還有許多其他變化,你可以在官方的Change Log 中獲取更多詳情碱鳞。不過(guò)桑李,我相信我已經(jīng)在本文涵蓋了主要的issues。

你可能會(huì)好奇現(xiàn)在是否是切換到Retrofit 2.0 的時(shí)機(jī)窿给?考慮到它仍然是beta階段贵白,你可能會(huì)希望繼續(xù)停留在1.9除非你跟我一樣是一個(gè)喜歡嘗鮮的人。 Retrofit 2.0用起來(lái)很好據(jù)我的經(jīng)驗(yàn)來(lái)看還沒(méi)有發(fā)現(xiàn)bug崩泡。

注意Retrofit 1.9 的官方文檔現(xiàn)在已經(jīng)從Square的github主頁(yè)刪除禁荒。我建議你現(xiàn)在就開(kāi)始學(xué)習(xí)Retrofit 2.0,盡快使用最新版本角撞。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末呛伴,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子靴寂,更是在濱河造成了極大的恐慌磷蜀,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,820評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件百炬,死亡現(xiàn)場(chǎng)離奇詭異褐隆,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)剖踊,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,648評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門庶弃,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)衫贬,“玉大人,你說(shuō)我怎么就攤上這事歇攻」坦撸” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,324評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵缴守,是天一觀的道長(zhǎng)葬毫。 經(jīng)常有香客問(wèn)我,道長(zhǎng)屡穗,這世上最難降的妖魔是什么贴捡? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,714評(píng)論 1 297
  • 正文 為了忘掉前任,我火速辦了婚禮村砂,結(jié)果婚禮上烂斋,老公的妹妹穿的比我還像新娘。我一直安慰自己础废,他們只是感情好汛骂,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,724評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著评腺,像睡著了一般帘瞭。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上歇僧,一...
    開(kāi)封第一講書(shū)人閱讀 52,328評(píng)論 1 310
  • 那天图张,我揣著相機(jī)與錄音,去河邊找鬼诈悍。 笑死祸轮,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的侥钳。 我是一名探鬼主播适袜,決...
    沈念sama閱讀 40,897評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼舷夺!你這毒婦竟也來(lái)了苦酱?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,804評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤给猾,失蹤者是張志新(化名)和其女友劉穎疫萤,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體敢伸,經(jīng)...
    沈念sama閱讀 46,345評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡扯饶,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,431評(píng)論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片尾序。...
    茶點(diǎn)故事閱讀 40,561評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡钓丰,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出每币,到底是詐尸還是另有隱情携丁,我是刑警寧澤,帶...
    沈念sama閱讀 36,238評(píng)論 5 350
  • 正文 年R本政府宣布兰怠,位于F島的核電站梦鉴,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏痕慢。R本人自食惡果不足惜尚揣,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,928評(píng)論 3 334
  • 文/蒙蒙 一涌矢、第九天 我趴在偏房一處隱蔽的房頂上張望掖举。 院中可真熱鬧,春花似錦娜庇、人聲如沸塔次。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,417評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)励负。三九已至,卻和暖如春匕得,著一層夾襖步出監(jiān)牢的瞬間继榆,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,528評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工汁掠, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留略吨,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,983評(píng)論 3 376
  • 正文 我出身青樓考阱,卻偏偏與公主長(zhǎng)得像翠忠,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子乞榨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,573評(píng)論 2 359

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

  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理秽之,服務(wù)發(fā)現(xiàn),斷路器吃既,智...
    卡卡羅2017閱讀 134,702評(píng)論 18 139
  • 相信很多人都在使用Retrofit考榨,我也在用,但是對(duì)它的理解都不是太深刻鹦倚,現(xiàn)在Retrofit2已經(jīng)出來(lái)一段時(shí)間河质,...
    WHOKNOWME閱讀 7,505評(píng)論 6 19
  • 前言 在Android開(kāi)發(fā)中,網(wǎng)絡(luò)請(qǐng)求十分常用 而在Android網(wǎng)絡(luò)請(qǐng)求庫(kù)中,Retrofit是當(dāng)下最熱的一個(gè)網(wǎng)...
    Carson帶你學(xué)安卓閱讀 70,767評(píng)論 48 393
  • 如果所有人都有了一個(gè)新的開(kāi)始愤诱。 新的開(kāi)始衍生最終的歸宿云头,我好像也只是又翻過(guò)了一篇心愛(ài)的小說(shuō)而已。 就像是我閱讀了你...
    Memi閱讀 496評(píng)論 0 0
  • 北京時(shí)間9月17日凌晨2點(diǎn)45淫半,2017-18賽季西甲第4輪溃槐,馬競(jìng)坐鎮(zhèn)萬(wàn)達(dá)大都會(huì)球場(chǎng)1-0小勝馬拉加,格列茲曼破門...
    英超球霸閱讀 291評(píng)論 0 2