在《Retrofit之請求參數(shù)》,我們討論了請求的url參數(shù)、表單編碼以及使用請求主體發(fā)送數(shù)據(jù)拆吆,但是處理API的請求我們還需要處理請求頭,來實現(xiàn)緩存脂矫、認證等操作枣耀。而在實際應用中,我們也通常會向請求頭添加平臺(Android or iOS)庭再、app版本捞奕、操作系統(tǒng)版本、網(wǎng)絡類型等信息拄轻。
定義請求頭
Retrofit提供了兩種方式來定義HTTP請求頭:靜態(tài)和動態(tài)颅围。
靜態(tài)請求頭
以獲取天氣的一個API接口為例,具體API詳見天氣查詢恨搓。這個接口請求要求一個apikey的請求頭以及一個citypinyi的url參數(shù)院促。
定義的接口如下:
public interface WeatherService {
@Headers("apikey:b86c2269fe6588bbe3b41924bb2f2da2")
@GET
Call<WeatherWrapper> weather(@Url String url, @Query("cityname") String cityName);
}
使用@Headers來定義請求頭的鍵值對,也可以同時定義多個請求頭斧抱,如下:
@Headers({
"key1:value1"常拓,
"key2:value2"
})
動態(tài)請求頭
動態(tài)請求頭以方法參數(shù)的形式存在,示例如下:
@GET
Call<WeatherWrapper> weather(@Header("apikey") String apikey, @Url String url, @Query("cityname") String cityName);
這樣辉浦,在發(fā)送請求(調(diào)用方法)時弄抬,就可以動態(tài)的定義apikey了。當然盏浙,這只是狹義上的動態(tài)眉睹,只能改變請求頭的值,而不能動態(tài)決定發(fā)送什么請求頭废膘。如果要動態(tài)的定義請求頭竹海,可是使用@HeaderMap。示例如下:
@GET
Call<WeatherWrapper> weather(@HeaderMap Map<String, String> headers, @Url String url, @Query("cityname") String cityName);
在OkHttp的攔截器中管理請求頭
在上面丐黄,我們了解了如何使用Retrofit定義請求頭斋配,回到文章開始我們所提及到的,我們通常會為每個API請求添加一些請求頭包含一些額外信息灌闺,如果在每個端點聲明都使用@Header來處理就麻煩了艰争。如果你知道在《Retrofit之請求參數(shù)》中如何為每個請求添加相同的url參數(shù),我會很感動的桂对。對甩卓,還是使用OkHttp的攔截器。對OkHttp攔截器不熟悉的蕉斜,可以參考《OkHttp之攔截器》逾柿。
從《OkHttp之示例》中的"訪問Header"中可以了解到缀棍,OkHttp提供了兩種方式來添加請求頭字段及值:你可以使用相同的key來重寫已存在的請求頭,或者單純添加而不管是否已有這樣的鍵值對存在机错。
重寫請求頭
Request.Builder中有個.header(key, value)的方法來允許我們定義請求頭爬范。如果這里之前已經(jīng)定義了一個相同鍵的請求頭,那么之前的則會被重寫弱匪。代碼示例如下:
okHttpClientBuilder.addInterceptor(new Interceptor() {
@Override
public Response intercept(Chain chain) throws IOException {
Request original = chain.request();
Request.Builder requestBuilder = original.newBuilder()
.header("token", "xxx")
.header("token", "yyy");
Request request = requestBuilder.build();
return chain.proceed(request);
}
});
最終請求頭token的值會從"xxx"重寫為"yyy"青瀑。如果應用中用戶登錄會持有一個token,之后需要加入到請求中萧诫,當token改變時(可能是更改密碼造成)斥难,此時token就應該更新,這時候就應該使用到重寫請求頭的方式财搁。
不重寫請求頭
如果說我們在定義請求頭而不想讓之前已定義的不被重寫蘸炸,從而兩個都存在,那么可以使用addHeader()方法替代header()方法尖奔,就是這么簡單,根據(jù)你的情況去合理使用即可穷当。
請求頭在項目中的應用
以我當前的項目來講提茁,會通過OkHttp攔截器的方式來添加請求頭,具體代碼示例如下:
okHttpClientBuilder.addInterceptor(new Interceptor() {
@Override
public Response intercept(Chain chain) throws IOException {
Request original = chain.request();
Request.Builder requestBuilder = original.newBuilder()
.header("platform", "platform")//平臺
.header("sysVersion", "sysVersion")//系統(tǒng)版本號
.header("device", "device")//設備信息
.header("screen", "screen")//屏幕大小
.header("uuid", "uuid")//設備唯一碼
.header("version", "version")//app版本
.header("apiVersion", "apiVersion")//api版本
.header("token", "token")//令牌
.header("channelId", "channelId")//渠道
.header("networkType", "networkType");//網(wǎng)絡類型
Request request = requestBuilder.build();
return chain.proceed(request);
}
});
至于你的項目馁菜,就看需求了茴扁。Ok,關(guān)于使用Retrofit定義請求頭就討論到這里汪疮。我們已經(jīng)討論了關(guān)于請求的url定義峭火、參數(shù)以及請求頭,接下來得討論下請求的發(fā)送智嚷、取消以及復用了卖丸。
源碼地址:
https://github.com/FILWAndroid/DevJourney
關(guān)于源碼:
- 不只是本文涉及的代碼,會包含很多知識點的代碼盏道,應該都會在我的簡書中進行介紹稍浆。
- 有可能代碼與本文中所貼出來的有差異,但應該都是我覺得更好的方式吧猜嘱。
- 新浪微博相關(guān)的代碼運行顯示不出來結(jié)果衅枫,感興趣的可以參考新浪微博SDK,配置工程朗伶。
- 歡迎大家對我進行批評教育弦撩。