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