After joining the company, the company required component-based development. After discussion, I packaged the network request framework separately, but at that time, the commonly used UTIl and UI in the framework were put into the Common package, resulting in some code coupling. In order to reduce the coupling, Common was split into lib_common and lib_ui. However, lib_UI relies on lib_common, which still leads to some code coupling. In the latest issue, in order to reduce the coupling between components, So I split the network requests in lib_common separately, and I made new packages and improvements, so the network framework is stable after three major changes.
Using the step
1. Initialize the Application class
ApiConfig build = new ApiConfig.Builder().setBaseurl (baseUrl)// baseUrl SetInvalidateToken (0)//Token invalidation code.setsucceedCode (200)// Success return code.setfilter ("com.mp5a5.quit.broadcastFilter"SetDefaultTimeout (2000)// Response time (2000 ms by default)//.setheads (headMap)// Dynamically added headers, You can also set //.setopenhttps (apiconfig.setheads ()) elsewheretrueSetSslSocketConfigure (sslSocketConfigure)//HTTPS authentication configuration.build (); build.init(this);Copy the code
2. Define interfaces
public interface NBAApiT {
@GET("onebox/basketball/nba")
Observable<NBAEntity> getNBAInfo(@QueryMap ArrayMap<String, Object> map);
}Copy the code
3. Create a request instance
Public class NbaService {private NBAApiT NBAApiT; privateNbaService() {
nbaApiT = RetrofitFactory.getInstance().create(NBAApiT.class);
}
public static NbaService getInstance() {
return Nbaservice1Holder.S_INSTANCE;
}
private static class Nbaservice1Holder {
private static final NbaService S_INSTANCE = new NbaService();
}
public Observable<NBAEntity> getNBAInfo(String key) {
ArrayMap<String, Object> map = new ArrayMap<>();
map.put("key", key);
returnnbaApiT.getNBAInfo(map); }}Copy the code
4. Send the request
findViewById(R.id.btnNBA).setOnClickListener(v -> {
NbaService.getInstance()
.getNBAInfo("6949e822e6844ae6453fca0cf83379d3") .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .compose(this.bindToLifecycle()) .subscribe(new BaseObserver<NBAEntity>(){ @Override public void onSuccess(NBAEntity response) { Toast.makeText(TestNBAActivity.this, response.result.title, Toast.LENGTH_SHORT).show(); }}); });Copy the code
5. Effect display
Packaging ideas
Returns the parameter callback
Since interfaces in JDK1.8 can have methods that do not need to be implemented by default, I used JDK1.8’s new features to encapsulate the callback of network request return parameters. The nice thing about this is that in some cases, we only need to deal with the requirements for success, but we don’t care much about failures and errors, so in the BaseObserver class I have a uniform wrapper around failures and errors so that we don’t have to deal with failures and errors every time we write callback arguments. Greatly reduces the amount of code.
public interface OnBaseResponseListener {
void onSuccess(R response);
default void onFailing(R response) {}
default void onError() {}}Copy the code
The onSuccess() method is mandatory. OnFailing (R Response) and onError() are optional. If your project wants to handle network request failures and errors, The onFailing(R Response) and onError() methods need to be overridden. If encapsulated loading boxes and Toast are used, Super.onfailing (response) and super.onError(e) are required, otherwise super can be omitted.
The entity-class Bean returned by the network request
This class is the parent of all network request entity classes. According to the code in this class, we determine whether the network request succeeds or fails or the token expires in the BaseObserver class.
public class BaseResponseEntity implements Serializable {
private static final long serialVersionUID = 1L;
public int code;
public String msg;
public boolean success() {
return ApiConfig.getSucceedCode() == code;
}
public int getTokenInvalid() {
returnApiConfig.getInvalidateToken(); }}Copy the code
For example, the entity class that requests NBA to return, we only need to inherit from BaseResponseEntity.
public class NBAEntity extends BaseResponseEntity {
@SerializedName("error_code") public int code; public String reason; public ResultBean result; public static class ResultBean { public String title; }}Copy the code
Since the code field returned by my project is not the code in BaseResponseEntity, I can alias it.
@SerializedName("error_code")
public int code;Copy the code
This will solve the problem that the code field returned by the company is not the same as the field I wrapped. Of course, it is not very friendly to write this code for each bean, so you can wrap another bean inherited from BaseResponseEntity and alias the code. Then the other beans in the project only need to inherit your own encapsulated beans.
The wrapper class BaseObserver for network state
This class inherits from the Observer class in RXJava. In this class, I judge the returned parameters in the onNext() method. If code is a successful code, the network request is successful; if code is not a successful code, the network request is failed. If the code returned is token invalid, I send a live broadcast. In my own project, you just need to receive the live broadcast in the base class of your activity, and then log out, clean up the data, etc. Also in the onError() method, I do the error handling. As I indicated above, the onFailing(Response) and onError(Throwable E) methods are not needed. Because I’ve done the same thing in this class.
Public abstract class BaseObserver<T extends BaseResponseEntity> implements Observer<T> {,,, @override public void onNext(T response) {if(response.success()) { try { onSuccess(response); } catch (Exception e) { e.printStackTrace(); }}else ifResponse.gettokeninvalid () == response.code. intent.setAction(ApiConfig.getQuitBroadcastReceiverFilter()); intent.putExtra(TOKEN_INVALID_TAG, QUIT_APP); AppContextUtils.getContext().sendBroadcast(intent); }else {
try {
onFailing(response);
} catch (Exception e) {
e.printStackTrace();
}
}
}
@Override
public void onError(Throwable e) {
if(e instanceof retrofit2.HttpException) {//HTTP error onException(exceptionReason.bad_network); }else if(e instanceof ConnectException | | e instanceof UnknownHostException) {/ / connection error onException(ExceptionReason.CONNECT_ERROR); }else if(e instanceof InterruptedIOException) {// Connection timeout onException(ExceptionReason.connect_timeout); }else if(e instanceof JsonParseException | | e instanceof JSONException | | e instanceof ParseException) {/ / parsing errors onException(ExceptionReason.PARSE_ERROR); }else{// Other errors onException(exceptionReason.unknown_error); }},,, @override public voidonComplete() {,,,} public void onSuccess(T response); public void onFailing(T response) { String message = response.msg;if (TextUtils.isEmpty(message)) {
Toast.makeText(AppContextUtils.getContext(), RESPONSE_RETURN_ERROR, Toast.LENGTH_SHORT).show();
} else{ Toast.makeText(AppContextUtils.getContext(), message, Toast.LENGTH_SHORT).show(); }},,}Copy the code
Retrofit encapsulation RetrofitFactory
This class is a wrapper around Retrofit in conjunction with OKTTP, Gson, interceptors, etc. Request timeouts, request header interceptors, request cache sizes, log interceptors, HTTPS authentication, return JSON processing, etc., are all handled in this class, which is arguably the core class for Retrofit.
Public class RetrofitFactory {,,, privateRetrofitFactory() {/ / specified path cache, the cache size to 100 MB File cacheFile = new File (AppContextUtils. GetContext () getCacheDir (),"HttpCache");
Cache cache = new Cache(cacheFile, 1024 * 1024 * 100);
OkHttpClient.Builder httpClientBuilder = new OkHttpClient().newBuilder()
.readTimeout(ApiConfig.getDefaultTimeout(), TimeUnit.MILLISECONDS)
.connectTimeout(ApiConfig.getDefaultTimeout(), TimeUnit.MILLISECONDS)
.addInterceptor(HttpLoggerInterceptor.getLoggerInterceptor())
.addInterceptor(new HttpHeaderInterceptor())
.addNetworkInterceptor(new HttpCacheInterceptor())
.cache(cache);
if(ApiConfig.getOpenHttps()) { httpClientBuilder.sslSocketFactory(1 == ApiConfig.getSslSocketConfigure().getVerifyType() ? SslSocketFactory.getSSLSocketFactory(ApiConfig.getSslSocketConfigure().getCertificateInputStream()) : SslSocketFactory.getSSLSocketFactory(), new UnSafeTrustManager()); httpClientBuilder.hostnameVerifier(new UnSafeHostnameVerify()); } OkHttpClient httpClient = httpClientBuilder.build(); Gson gson = new GsonBuilder() .setDateFormat("yyyy-MM-dd HH:mm:ss")
.serializeNulls()
.registerTypeAdapterFactory(new NullTypeAdapterFactory())
.create();
retrofit = new Retrofit.Builder()
.client(httpClient)
.addConverterFactory(GsonConverterFactory.create(gson))
.addCallAdapterFactory(RxJava2CallAdapterFactory.create());
if(! TextUtils.isEmpty(ApiConfig.getBaseUrl())) { build = retrofit.baseUrl(ApiConfig.getBaseUrl()).build(); }},,, public <T> T create(Class<T> clazz) {checkNotNull(build,"BaseUrl not init,you should init first!");
return build.create(clazz);
}
public <T> T create(String baseUrl, Class<T> clazz) {
returnretrofit.baseUrl(baseUrl).build().create(clazz); }}Copy the code
Use the configuration class ApiConfig
This class is where all the initialization parameters are configured, such as return code, BaseUrl, invalid InvalidateToken, request header Heads, whether HTTPS authentication is enabled, etc. You can configure the parameters you need in the application of the project, so that the project only needs to write the code related to the request, and there is no need to deal with the tedious tasks such as request baseUrl and return code.
public class ApiConfig implements Serializable { private static int mInvalidateToken; private static String mBaseUrl; ,, private ApiConfig(Builder Builder) {mInvalidateToken = Builder. InvalidateToken; mBaseUrl = builder.baseUrl; } public void init(Context appContext) {appContextutils.init (appContext); } public static intgetInvalidateToken() {
return mInvalidateToken;
}
public static String getBaseUrl() {
returnmBaseUrl; },,, public static final Class Builder {private int invalidateToken; private String baseUrl; Public BuildersetBaseUrl(String mBaseUrl) {
this.baseUrl = mBaseUrl;
return this;
}
public Builder setInvalidateToken(int invalidateToken) {
this.invalidateToken = invalidateToken;
returnthis; },,, public ApiConfigbuild() {
returnnew ApiConfig(this); }}}Copy the code
This article is not easy, if you like this article, or helpful to you hope you more, like, forward, follow oh. The article will be updated continuously. Absolutely dry!!
Author: Mp5A5
Link: https://www.jianshu.com/p/181227ca8a4d