MVP实战心得---封装Retrofit2

本文详细介绍了如何使用RxJava和Retrofit进行网络请求,包括注解的使用、Retrofit的配置、数据解析工厂的设置,以及如何通过Retrofit+RxAndroid实现请求的封装和错误处理。作者还给出了学习MaterialDesign控件的建议和对NDK与JVM基础知识的看法。
摘要由CSDN通过智能技术生成

传送门:RxJava---------这个作为入门学习rxjava非常好

2.Retrofit

这个写点基本的用法吧…

首先看用的包:

//retrofit2–看名字就知道是啥了
compile ‘com.squareup.retrofit2:retrofit:2.1.0’
//CallAdapterFactory的Rx依赖包—导这个包才能配合rxAndroid使用
compile ‘com.squareup.retrofit2:adapter-rxjava:2.1.0’
//ConverterFactory的String依赖包----这个是解析数据的工厂.用来格式化数据的,配置编码啊,gson解析啊.
compile ‘com.squareup.retrofit2:converter-scalars:2.1.0’

然后是retrofit注解:(使用retrofit,注解是很重要的)

方法注解 : 包含@GET、@POST、@PUT、@DELETE、@PATCH、@HEAD、@OPTIONS、@HTTP。

这个不多讲.一般用的就是@GET、@POST,很明显,一个是get请求,一个是post请求

标记注解 : 包含@FormUrlEncoded、@Multipart、@Streaming。
这个得和参数注解一起说

参数注解 : 包含@Query,@QueryMap、@Body、@Field,@FieldMap、@Part,@PartMap。

@Get---------用的参数注解就@Query,@QueryMap,

@Post--------则会用到 @Body、@Field,@FieldMap、@Part,@PartMap。

@Body-------将数据转化成Json,然后post.具体转化根据设置的解析工厂(下面有讲)
---------------------------------------------------分割线----------------------------------------------------------
@Field,@FieldMap------post上传表单.@Field表示单个,@FieldMap表示集合.
需要添加上面的@FormUrlEncoded表示表单提交 ,
对应Content-Type:application/x-www-form-urlencoded
如:
@FormUrlEncoded
@POST(“login的url”)
Observable login(@Field(“name”) String name, @FieldMap Map params);
--------------------------------------------------分割线------------------------------------------------------------
@Part,@PartMap----post上传文件/数据.@Part表示单个,@PartMap表示集合.
其中@Part MultipartBody.Part 类型代表文件,@Part(“key”) RequestBody类型代表参数 
需要添加@Multipart表示支持文件上传的表单,Content-Type: multipart/form-data
@Multipart
@POST(“update的url”)
Observable update(@Part (“file”) MultipartBody.Part file, @Part(“key”) RequestBody key,@PartMap Map<String,RequestBody> files);
如果参数较少,使用@Part (“file”)就可以解决了,如果参数较多,那就需要使用@PartMap了.

其他注解 : @Path、@Header,@Headers、@Url
这几个用处挺大的,这里就不细说了,并不是必用的,我用的不多.

Retrofit 配置代码.

//这个是处理网络请求的log信息的,可以实现Interceptor接口来自定义.
HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor(new HttpLoggingInterceptor.Logger() {
@Override
public void log(String message) {
HLog.i(“RxJava”, message);
}
});

OkHttpClient client = new OkHttpClient.Builder()
.addInterceptor(interceptor)
.build();
Retrofit retrofit = new Retrofit.Builder()
.client(client)//Retrofit需要配置一个OkHttpClient实例.
.baseUrl(API_HOST)//需要指定一个baseUrl,一般就是服务器的域名
.addConverterFactory(FastjsonConverterFactory.create())//这个是数据解析工厂,我用的是fastjson
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())//支持rxJava,在第二个jar包里面
.build();

下面是完整代码:

/**

  • 写成单例模式,因为并不需要多个Retrofit存在.
    /
    public class RetrofitUtil {
    /
    *
  • 服务器地址
    */
    private static final String API_HOST =“你的BaseUrl”;
    private RetrofitUtil() {

}
public static Retrofit getRetrofit() {
return Instanace.retrofit;
}
//静态内部类,保证单例并在调用getRetrofit方法的时候才去创建.
private static class Instanace {
private static final Retrofit retrofit = getInstanace();
private static Retrofit getInstanace() {
HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor(new HttpLoggingInterceptor.Logger(){
@Override
public void log(String message) {
HLog.i(“RxJava”, message);
}
});
OkHttpClient client = new OkHttpClient.Builder()
.addInterceptor(interceptor)
.build();
Reretrofit = new Retrofit.Builder()
.client(client)
.baseUrl(API_HOST)
.addConverterFactory(FastjsonConverterFactory.create())
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.build();
return retrofit;
}
}
}

json解析工厂,代码太多,具体可以见demo

3.RxAndroid

如果没接触的话,可以看前面的Rxjava链接.

(1).首先看Reretrofit+RxAndroid是怎么使用的

@GET(“login地址”)
Observable<BaseResponse> login(@QueryMap HashMap<String, Object> params);

其实所谓的Reretrofit+RxAndroid就是这么回事.

没有RxAndroid的Reretrofit请求接口是这样写的:

@GET(“login地址”)
Call<BaseResponse> login(@QueryMap HashMap<String, Object> params);

把Call换成了Observable而已.

(2)写一个接口类

/**
*所有的网络请求都可以写在这个接口类里面.
*/
public interface APIService {
@GET(“login地址”)
Observable<BaseResponse> login(@QueryMap HashMap<String, Object> params);

}

(3)接口类的实现

/**

  • 请求生成类。Retrofit一次生成,并作为单例.
    */
    public class ApiServcieImpl {
    private ApiServcieImpl() {

}
public static APIService getInstance() {
return createAPIService.apiService;
}

/**

  • Retrofit生成接口对象.
    */
    private static class createAPIService {
    //Retrofit会根据传入的接口类.生成实例对象.
    private static final APIService apiService = RetrofitUtil.getRetrofit().create(APIService.class);
    }
    }

然后就可以通过ApiServcieImpl.getInstance()去调用APIService里面写的接口了. 如:

ApiServcieImpl.getInstance().login(new HashMap<String, Object>()) //传入参数
.subscribe(new Action1<BaseResponse>() {//简单的回调
@Override
public void call(BaseResponse loginDataBaseResponse) {
//拿到数据,做处理
}
});

4.封装

有没有发现,设置泛型的是,包了一层BaseResponse 这么做是为了请求完成后对返回的数据进行统一处理.

先看BaseResponse:

public class BaseResponse {
private boolean success;//请求是否成功
private int resultCode;//状态吗
private String msg;//返回的提示消息
private T data;//主要内容,因为不知道返回的会是什么类型,所以用泛型来表示
//get set方法就不贴了.
}

怎么处理.

/**

  • @author jlanglang 2016/11/15 16:14
    */
    public class ModelFilteredFactory {
    private final static Observable.Transformer transformer = new SimpleTransformer();

/**

  • 将Observable<BaseResponse>转化Observable,并处理BaseResponse
  • @return 返回过滤后的Observable.
    */
    @SuppressWarnings(“unchecked”)
    public static Observable compose(Observable<BaseResponse> observable) {
    return observable.compose(transformer);
    }

/**

  • 这里就不细讲了,具体可以去看rxjava的使用.这个类的意义就是转换Observable.
    */
    private static class SimpleTransformer implements Observable.Transformer<BaseResponse, T> {
    //这里对Observable,进行一般的通用设置.不用每次用Observable都去设置线程以及重连设置
    @Override
    public Observable call(Observable<BaseResponse> observable) {
    return observable.subscribeOn(Schedulers.io())
    .observeOn(AndroidSchedulers.mainThread())
    .unsubscribeOn(Schedulers.io())
    .timeout(5, TimeUnit.SECONDS)//重连间隔时间
    .retry(5)//重连次数
    .flatMap(new Func1<BaseResponse, Observable>() {
    @Override
    public Observable call(BaseResponse tBaseResponse) {
    return flatResponse(tBaseResponse);
    }
    });
    }

/**

  • 处理请求结果,BaseResponse
  • @param response 请求结果
  • @return 过滤处理, 返回只有data数据的Observable
    */
    private Observable flatResponse(final BaseResponse response) {
    return Observable.create(new Observable.OnSubscribe() {
    @Override
    public void call(Subscriber<? super T> subscriber) {
    if (response.isSuccess()) {//请求成功
    if (!subscriber.isUnsubscribed()) {
    subscriber.onNext(response.getData());
    }
    } else {//请求失败
    int resultCode = response.getResultCode();
    if (!subscriber.isUnsubscribed()) {
    //这里抛出自定义的一个异常.可以处理服务器返回的错误.
    subscriber.onError(new APIException(response.getResultCode(), response.getMsg()));
    }
    return;
    }
    if (!subscriber.isUnsubscribed()) {//请求完成
    subscriber.onCompleted();
    }
    }
    });
    }
    }
    }

仅仅只有上面那些了么?接着看.

/**

  • @author jlanglang 2016/11/14 17:32
  • Subscriber,这个是用来处理Observable的结果的.
    */
    public abstract class SimpleSubscriber extends Subscriber {

@Override
public void onCompleted() {//这个是请求完成时调用.如果走了onError()就不会走这个方法.

}

@Override
public void onError(Throwable e) {//这里通常就处理异常
if (e instanceof APIException) {
APIException exception = (APIException) e;
ToastUtil.showToast( exception.message);
} else if (e instanceof UnknownHostException) {
ToastUtil.showToast(“请打开网络”);
} else if (e instanceof SocketTimeoutException) {
ToastUtil.showToast( “请求超时”);
} else if (e instanceof ConnectException) {
ToastUtil.showToast(“连接失败”);
} else if (e instanceof HttpException) {
ToastUtil.showToast(“请求超时”);
}else {
ToastUtil.showToast(“请求失败”);
}
e.printStackTrace();
}

@Override
public void onNext(T t) {//这里的是获得了数据,方法意思很明显,下一步干啥
if (t != null) {//这里最好判断一下是否为null.
call(t);
} else {
ToastUtil.showToast(“连接失败”);
}
}
/**
*因为具体的处理这里无法得知,所以抽象.
*/
public abstract void call(T t);
}

好了,看看现在的具体使用吧:

ModelFilteredFactory.compose(ApiServcieImpl.getInstance().login(new HashMap<String, Object>()))
.subscribe(new SimpleSubscriber() {
@Override
public void call(LoginData loginData) {

}
});
看起来之前用起来差不多,但是却做了很多的处理:
1.对Observable做了通用设置.网络重连次数,线程设置,重连时间.
2.做了对服务器返回结果的统一处理.比如根据resultcode,处理登陆过期啊啥的.
3.判断了data是否为null,不会在call()里面担心loginData是否为null

尾声

评论里面有些同学有疑问关于如何学习material design控件,我的建议是去GitHub搜,有很多同行给的例子,这些栗子足够入门。

有朋友说要是动真格的话,需要NDK以及JVM等的知识,首现**NDK并不是神秘的东西,**你跟着官方的步骤走一遍就知道什么回事了,无非就是一些代码格式以及原生/JAVA内存交互,进阶一点的有原生/JAVA线程交互,线程交互确实有点蛋疼,但平常避免用就好了,再说对于初学者来说关心NDK干嘛,据鄙人以前的经历,只在音视频通信和一个嵌入式信号处理(离线)的两个项目中用过,嵌入式信号处理是JAVA->NDK->.SO->MATLAB这样调用的我原来MATLAB的代码,其他的大多就用在游戏上了吧,一般的互联网公司会有人给你公司的SO包的。
至于JVM,该掌握的那部分,相信我,你会掌握的,不该你掌握的,有那些专门研究JVM的人来做,不如省省心有空看看计算机系统,编译原理。

一句话,平常多写多练,这是最基本的程序员的素质,尽量挤时间,读理论基础书籍,JVM不是未来30年唯一的虚拟机,JAVA也不一定再风靡未来30年工业界,其他的系统和语言也会雨后春笋冒出来,但你理论扎实会让你很快理解学会一个语言或者框架,你平常写的多会让你很快熟练的将新学的东西应用到实际中。
初学者,一句话,多练。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值