現(xiàn)在都在說Retrofit2和RxJava2,作為一個(gè)程序猿自然不能落伍,然后就試用了一下,遇到一些坑,在這里記錄一下
1.
io.reactivex.exceptions.OnErrorNotImplementedException: closed
和
Caused by: java.lang.IllegalStateException: closed
這兩個(gè)異常今天研究了好幾個(gè)小時(shí)... 網(wǎng)上都是說Response.body().string()方法不能調(diào)用2次,但是在我的代碼里這個(gè)方法并沒有調(diào)用2次,一個(gè)Response的ResponseBody屬性我只調(diào)用了1次string()方法,后來終于發(fā)現(xiàn),在我的判斷里有不同的流程,其中一個(gè)流程就沒有問題,然后發(fā)現(xiàn)流程A的Response我直接return了,但是流程B的Response我調(diào)用了它的body().string()方法,我判斷了里面的返回結(jié)果字符串,發(fā)現(xiàn)如果沒有錯(cuò)誤就把這個(gè)流程B的Response對(duì)象return了,但是!!! 因?yàn)槲襯eturn的這個(gè)Response已經(jīng)被我調(diào)用過body().string()方法了,Retrofit2把結(jié)果返回上層應(yīng)用的時(shí)候,在convert的時(shí)候(因?yàn)閞etrofit調(diào)用了addConverterFactory方法)就出錯(cuò)了,因?yàn)檫@個(gè)Response的ResponseBody屬性已經(jīng)close了,所以在返回的時(shí)候,需要一個(gè)新的Response,并設(shè)置其中的body為新的body,代碼如下:
Response response = chain.proceed(newRequest);//執(zhí)行新請(qǐng)求
String responseString = response.body().string();
//對(duì)responseString...執(zhí)行一些判斷之類的操作
return response.newBuilder().body(ResponseBody.create(response.body().contentType(), responseString)).build();
注意:
1.上面的這個(gè)問題,如果你添加了自己的log攔截器,在你的log攔截器里如果調(diào)用了Response的body屬性的string()方法,那么返回的response對(duì)象你也要這么處理,否則在其他類里時(shí)候的時(shí)候,雖然response不是一個(gè)對(duì)象,但是里面的body是同一個(gè)對(duì)象(不信的可以debug查看log攔截器里面你返回的response的body屬性內(nèi)存地址和其他攔截器里同請(qǐng)求的這個(gè)response的body屬性的內(nèi)存地址),然后其他類調(diào)用response.body().string()的時(shí)候會(huì)報(bào)close錯(cuò)誤,所以只要用到了response.body().string()方法,在返回這個(gè)response的時(shí)候就要像上面那樣返回才可以避免close問題
2.如果不是string()方法,你用的是response.body().bytes()方法再轉(zhuǎn)字符串是一樣的,在方法內(nèi)部都close了,所以仍然需要像上面那樣處理
2.
Caused by: java.lang.IllegalStateException: network interceptor xxx must call proceed() exactly once
和
io.reactivex.exceptions.OnErrorNotImplementedException: network interceptor xxx must call proceed() exactly once
這是因?yàn)槟愕?code>OkHttpClient.Builder調(diào)用的addNetworkInterceptor方法添加的攔截器,這種方式添加的攔截器里面的Chain只能調(diào)用一次proceed方法,如果想調(diào)用多次,你的攔截器就要用addInterceptor方法添加,就沒問題了