OkHttp can be said to be the Android development, every project must rely on the network library, we can be very convenient and efficient processing network requests, greatly improve the coding efficiency. But sometimes we run into this problem with OkHttp
Stacktrace crashes
E AndroidRuntime: FATAL EXCEPTION: OkHttp Dispatcher
E AndroidRuntime: Process: com.example.okhttpexceptionsample, PID: 13564
E AndroidRuntime: java.lang.NullPointerException: blablabla
E AndroidRuntime: at com.example.okhttpexceptionsample.MainActivity$createNPEInterceptorThe $1.intercept(MainActivity.kt:61)
E AndroidRuntime: at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:112)
E AndroidRuntime: at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:87)
E AndroidRuntime: at okhttp3.RealCall.getResponseWithInterceptorChain(RealCall.kt:184)
E AndroidRuntime: at okhttp3.RealCall$AsyncCall.run(RealCall.kt:136)
E AndroidRuntime: at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
E AndroidRuntime: at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
E AndroidRuntime: at java.lang.Thread.run(Thread.java:784)
Copy the code
Two. Why did it crash
From the stackTrace above, we can analyze that a NullPointerException has occurred. There was a crash.
Wait, I remember OkHttp handled exceptions.
Well, yes, OkHttp does handle exceptions, such as onFailure. For example, the following describes the content of Callback.
interface Callback {
/**
* Called when the request could not be executed due to cancellation, a connectivity problem or
* timeout. Because networks can fail during an exchange, it is possible that the remote server
* accepted the request before the failure.
*/
fun onFailure(call: Call, e: IOException)
/**
* Called when the HTTP response was successfully returned by the remote server. The callback may
* proceed to read the response body with [Response.body]. The response is still live until its
* response body is [closed][ResponseBody]. The recipient of the callback may consume the response
* body on another thread.
*
* Note that transport-layer success (receiving a HTTP response code, headers and body) does not
* necessarily indicate application-layer success: `response` may still indicate an unhappy HTTP
* response code like 404 or 500.
*/
@Throws(IOException::class)
fun onResponse(call: Call, response: Response)
}
Copy the code
Yes.
- OkHttp handles only IOException cases,
- NullPointerException is not a subclass of IOException
So it doesn’t get processed, it crashes.
So is there a way to fix it so that this crash doesn’t happen and doesn’t interfere with users? Well, you can.
Use Interceptor
package com.example.okhttpexceptionsample import okhttp3.Interceptor import okhttp3.Response import java.io.IOException /** * The Throwable that may occur in the intercept of the Interceptor is wrapped in IOExceptionWrapper, which is converted to a network request failure instead of an application crash */ class SafeGuardInterceptor: Interceptor { override fun intercept(chain: Interceptor.Chain): Response { try {return chain.proceed(chain.request())
} catch (t: Throwable) {
throw IOExceptionWrapper("SafeGuarded when requesting ${chain.request().url}", t)}}} /** * Wrap the Throwable that occurred in chain.proceed as IOExceptionWrapper */ class IOExceptionWrapper(message: String? , cause: Throwable?) : IOException(message, cause)Copy the code
In the above code, we converted any Throwable into IOExceptionWrapper (disguised as IOException) and added it to OkHttpClient
fun createOKHttpClient(): OkHttpClient {
return OkHttpClient.Builder()
.addInterceptor(SafeGuardInterceptor())
.build()
}
Copy the code
The next time we execute the code with the NPE, the log changes (no longer crash logs, but exceptions)
W System.err: com.example.okhttpexceptionsample.IOExceptionWrapper: SafeGuarded=blablabla
W System.err: at com.example.okhttpexceptionsample.SafeGuardInterceptor.intercept(SafeGuardInterceptor.kt:12)
W System.err: at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:112)
W System.err: at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:87)
W System.err: at okhttp3.RealCall.getResponseWithInterceptorChain(RealCall.kt:184)
W System.err: at okhttp3.RealCall$AsyncCall.run(RealCall.kt:136)
W System.err: at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
W System.err: at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
W System.err: at java.lang.Thread.run(Thread.java:784)
W System.err: Caused by: java.lang.NullPointerException: blablabla
W System.err: at com.example.okhttpexceptionsample.MainActivity$createNPEInterceptorThe $1.intercept(MainActivity.kt:61)
W System.err: at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:112)
W System.err: at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:87)
W System.err: at com.example.okhttpexceptionsample.SafeGuardInterceptor.intercept(SafeGuardInterceptor.kt:10)
W System.err: ... 7 more
Copy the code
Two points need to be noted above
- Adding an Interceptor instead of a NetworkInterceptor
- Order is important, always put it first
Four. What’s wrong with that
This, of course, significantly increases the stability of requests and the crash rate of applications. But are there some problems? Such as
- Yes, the above problems may exist, but we can mitigate or solve the problems in the following ways
- Use SafeGuardInterceptor only for release cases, which makes it easier to find in DEBUG cases
- Variants will be configurable for different Build variants, so that problems can be found in a small range as possible
- Implement a more intelligent dynamic opening strategy.
In software engineering, many decisions are the embodiment of trade-off, and the specific implementation plan can be balanced by everyone.
About me
More Android advanced interview collection on Github, you can click about me contact me to get very hope to communicate with you, common progress
At present, I am a programmer. I not only share the knowledge related to Android development, but also share the growth history of technical people, including personal summary, workplace experience, interview experience, etc., hoping to make you take a little detours.