1. 程式人生 > >MVP實戰心得(三)---封裝Retrofit2.0+RxAndroid+RxBus

MVP實戰心得(三)---封裝Retrofit2.0+RxAndroid+RxBus

介紹:

Retrofit:

對okhttp的封裝,可以更方便的使用okhttp

RxAndroid

響應式程式設計框架,rxjava的擴充套件,很爽的鏈式程式設計 魅力在於對資料的處理,與執行緒切換的靈活性. 用來處理非同步操作(Lambda表示式不會用.用Lambda表示式程式碼會更少,但不會的人會看不懂程式碼.不是很推薦)

RxBus

用RxJava實現的EventBus

說說為什麼要配合起來用

Retrofit負責連結網路,請求網路. RxAndroid負責處理請求的結果.非同步操作 RxBus可以很方便的進行各元件之間的通訊. 我之前是用asynchttpclient做網路請求的,各種程式碼縮排,if套if,各種回撥,慘不忍睹啊. 用了Retrofit+RxAndroid我就徹底放棄asynchttpclient了.

使用

1.RxJava

傳送門: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上傳表單
[email protected]
表示單個,@FieldMap表示集合. 需要新增上面的@FormUrlEncoded表示表單提交 , 對應Content-Type:application/x-www-form-urlencoded 如: @FormUrlEncoded @POST("login的url") Observable<User> login(@Field("name") String name, @FieldMap Map params); --------------------------------------------------分割線------------------------------------------------------------ @Part,@PartMap----post上傳檔案/資料[email protected]表示單個,@PartMap表示集合. 其中@Part MultipartBody.Part 型別代表檔案,@Part(“key”) RequestBody型別代表引數 需要新增@Multipart表示支援檔案上傳的表單,Content-Type: multipart/form-data @Multipart @POST("update的url") Observable<User> 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<LoginData>> login(@QueryMap HashMap<String, Object> params);

其實所謂的Reretrofit+RxAndroid就是這麼回事.

沒有RxAndroid的Reretrofit請求介面是這樣寫的:

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

把Call換成了Observable而已.

(2)寫一個介面類
/**
*所有的網路請求都可以寫在這個介面類裡面.
*/
public interface APIService {
     @GET("login地址")
     Observable<BaseResponse<LoginData>> 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<LoginData>>() {//簡單的回撥
            @Override 
             public void call(BaseResponse<LoginData> loginDataBaseResponse) { 
                //拿到資料,做處理
            }
    });

4.封裝

有沒有發現,設定泛型的是<BaseResponse<LoginData>>,包了一層BaseResponse
這麼做是為了請求完成後對返回的資料進行統一處理.

先看BaseResponse:

public class BaseResponse<T> {
    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<T>>轉化Observable<T>,並處理BaseResponse
     *
     * @return 返回過濾後的Observable.
     */
    @SuppressWarnings("unchecked")
    public static <T> Observable<T> compose(Observable<BaseResponse<T>> observable) {
        return observable.compose(transformer);
    }

    /**
     * 這裡就不細講了,具體可以去看rxjava的使用.這個類的意義就是轉換Observable.
     */
    private static class SimpleTransformer<T> implements Observable.Transformer<BaseResponse<T>, T> {
        //這裡對Observable,進行一般的通用設定.不用每次用Observable都去設定執行緒以及重連設定
        @Override
        public Observable<T> call(Observable<BaseResponse<T>> observable) {
            return observable.subscribeOn(Schedulers.io())
                      .observeOn(AndroidSchedulers.mainThread())
                      .unsubscribeOn(Schedulers.io())
                      .timeout(5, TimeUnit.SECONDS)//重連間隔時間
                      .retry(5)//重連次數
                      .flatMap(new Func1<BaseResponse<T>, Observable<T>>() {
                          @Override
                          public Observable<T> call(BaseResponse<T> tBaseResponse) {
                              return flatResponse(tBaseResponse);
                          }
                      });
        }

        /**
         * 處理請求結果,BaseResponse
         * @param response 請求結果
         * @return 過濾處理, 返回只有data資料的Observable
         */
        private Observable<T> flatResponse(final BaseResponse<T> response) {
            return Observable.create(new Observable.OnSubscribe<T>() {
                @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<T> extends Subscriber<T> {

    @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<LoginData>() {
                    @Override
                    public void call(LoginData loginData) {

                    }
                });
 看起來之前用起來差不多,但是卻做了很多的處理:
1.對Observable做了通用設定.網路重連次數,執行緒設定,重連時間.
2.做了對伺服器返回結果的統一處理.比如根據resultcode,處理登陸過期啊啥的.
3.判斷了data是否為null,不會在call()裡面擔心loginData是否為null
4.統一處理了請求的各種異常.

5.用到MVP中.

你以為上面那些就完了嗎?NO!
如果我們在Presenter中這樣呼叫其實是很不科學的.
ModelFilteredFactory.compose(ApiServcieImpl.getInstance().login(new HashMap<String, Object>()))
這個轉換我們應該放在Modle和ModleImpl中去寫
public class LoginContract{
     ....//view介面省略
    public interface Model {  
      /**   
      * 獲取登陸資料  
      * @return Observable<LoginData> 
      */  
      Observable<LoginData> login(HashMap<String, Object> treeMap);
    }
    ....//prensent介面省略
}

public class LoginModelImpl implements LoginContract.Model { 
   @Override 
   public Observable<LoginData> login(HashMap<String, Object> hashMap) {     
       return ModelFilteredFactory.compose(ApiServcieImpl.getInstance().login(hashMap));   
 }}
那麼我們在presenter中呼叫就可以這樣:
public class LoginPresenterImpl exdents BasePresenter implements LoginContract.Presenter{
     .....
     private  LoginModelImpl  loginModelImpl;
     public void onCreate(){
       loginModelImpl = new LoginModelImpl();//建立modle例項
     }
     public void login(){
        //通過modle請求介面
        loginModelImpl.login(new HashMap<String, Object>()))
                .subscribe(new SimpleSubscriber<LoginData>() {
                    @Override
                    public void call(LoginData loginData) {
                          //處理請求的資料,繫結檢視 
                    }
                });
        }
    ....
}

6.管理Observable的生命週期,也就是網路請求的生命週期.

Observable是不是很高大上,然而如果你不進行處理,可是會記憶體洩漏的
RxAndroid也不會自動的根據Activity/frgament的生命週期結束非同步請求.
但處理其實很簡單.

使用CompositeSubscription

只需要將Observable,非同步處理到最後返回的subscribe新增到CompositeSubscription例項裡就行了.

  public void login(){
    Subscription subscribe = loginModelImpl.login(new HashMap<String, Object>()))
                .subscribe(new SimpleSubscriber<LoginData>() {
                    @Override
                    public void call(LoginData loginData) {
                          //處理請求的資料,繫結檢視 
                    }
                });
      compositeSubscription.add(subscribe);//新增訂閱
  }
  //在銷燬的時候,結束訂閱事件.
  public void onDestroy() {
    compositeSubscription.unsubscribe();//結束所有add的subscribe事件
 }

那麼,實戰心得(二)中的BasePresenter就可以進行改進了,具體見:

/**
 * @author jlanglang  2016/11/11 15:10
 */
public abstract class BasePresenter<T extends BaseView> {
   protected T mView;
   protected CompositeSubscription compositeSubscription;
    /**
     * 繫結View
     */
    public void onAttch(T view) {
        this.mView = view;
        compositeSubscription = new CompositeSubscription ();
    }
    /**
     * 做初始化的操作,需要在V的檢視初始化完成之後才能呼叫
     * presenter進行初始化.
     */
    public abstract void onCreate();
    /** 
    * 在這裡結束非同步操作
    */
    public void onDestroy(){
        compositeSubscription.unsubscribe();//結束非同步請求.
    }
    /**
     * 在V銷燬的時候呼叫,解除繫結
    */
    public void onDetach() {  
       mView = null;
    }
    /**
    * 容易被回收掉時儲存資料
    */
    public abstract void onSaveInstanceState(Bundle outState);
}

7 RxBus

沒什麼特別值得提的,用法自行搜尋,哈哈,個人在專案中用的也不是很多,某些情況會用一下,但真心好用.

再附上githubdemo地址,還未更新到最新.

相關推薦

MVP實戰心得()---封裝Retrofit2.0+RxAndroid+RxBus

介紹: Retrofit: 對okhttp的封裝,可以更方便的使用okhttp RxAndroid 響應式程式設計框架,rxjava的擴充套件,很爽的鏈式程式設計 魅力在於對資料的處理,與執行緒切換的靈活性. 用來處理非同步操作(Lambda表示式不會用.用Lambda表示式程式碼會更少,但不會的人會看不懂

Mvp實戰心得(二)---Base基類的封裝

基類: 在開發中有著很重要的地位,基類設計封裝的好,不管是對app的效能,還是對開發的效率都有著很重要的作用 基類可以簡單幾個字概況,一個良好的父類. 結構: 不管你的app是多個acitivity,還是1個activity+n個fragment,還是多個acitivity多個fragment. 始終都是用

RxJava + Retrofit2.0的專案實戰完美封裝

Retrofit 和RxJava已經出來很久了,從去年開始rxjava和retrofit就開始火,所以之前在做專案的時候也用了rxjava和retrofit,今天就介紹一下在專案中如何封裝rxjava和retrofit。對於 RxJava 不是很瞭解的同學推薦你

Android:這是一份非常詳細的MVP+Rxjava2.0+Retrofit2.0相結合舉例RecyclerView的實戰篇章

MVP+Rxjava2.0+Retrofit2.0現在是非常火的組合 MVP相信大家已經在各大網站和各大佬的文章中已經瞭解很多理論的理解了 MVP其實就是M層請求資料 在P層裡進行M層和V層的互動 V層得到資料後展示資料 比如說豺狼媽媽去捕食 捕到食物後

專案中關於Retrofit2.0+RxJava+OkHttp的封裝和使用

1.引入依賴庫 在app的 build.gradle檔案中新增如下配置: //下面兩個是RxJava 和RxAndroid compile 'io.reactivex:rxandroid:1.2.1' compile 'io.reactivex:rxja

Android 基於Retrofit2.0的支援多主機地址的網路請求類的封裝

一、首先在Module級別的build.gradle檔案中新增依賴 implementation 'com.squareup.retrofit2:retrofit:2.3.0' implementation 'com.squareup.retrofit2:adapter-r

Retrofit2.0和RxJava2.0的簡單封裝

一、首先新增專案依賴: implementation "com.squareup.okhttp3:logging-interceptor:$var.loggingInterceptor"

Retrofit2.0 從基礎到實戰(一)Retrofit的使用方法

前言 Retrofit2.0的基礎就是okhttp,現在多與RxJava一起使用,恩,下面先講一下它的優缺點: 優點: 可以配置不同HTTP client來實現網路請求,如okhttp、httpclient等。 支援同步、非同步和RxJava

Android基於Retrofit2.0 +RxJava 封裝的超好用的RetrofitClient工具類(六)

RetrofitClient 避免重複建立Retrofit實列. 呼叫方便簡潔. 無需重複設定屬性的步驟. 可固定配置 Host 也可動態配置Url、請求頭、引數等. 支援檔案下載和上傳. 可支援泛型擴充套件的ApiService 支

從零開始搭建一個主流專案框架()—RxJava2.0+Retrofit2.0+OkHttp

個人部落格:haichenyi.com。感謝關注   上一篇,我們把mvp+dagger加進去了,這一篇,我們把網路請求加上   我這裡的網路請求是用的裝飾者模式去寫的,什麼是裝飾者模式呢?在不必改變原類檔案和使用繼承的情況下,動態地擴充套件一個物件的功能。

rxjava+rxandroid+retrofit2.0使用方法demo講解

最近看了retrofit框架,比較新的網路訪問請求框架,很多專案都涉及到,所以學習了一下,做個總結。 第一講 retrofit2.0單獨使用 1、匯入依賴jar包(也可能是library,反正studio會自動下載) compile ‘io.react

Android基於Retrofit2.0+RxJava 封裝的超好用的工具類

/** * Created by Tamic on 2016-07-08. */ public interface ApiService { public static final String Base_URL = "http://ip.taobao.com/"; /** *普通寫法 */ @

android中Retrofit2.0封裝:設計到請求前後的操作,比如新增請求頭,攔截請求頭,攔截返回體等

這裡關於android如何整合retrofit2.0就詳細介紹了,相信網路上也有很多的例子。首先retrofit關於請求體如何加入的話,這裡使用註解就能解決大部分問題,而retrofit官網也給出了很多很好用的註解,只要在interface中宣告就行。主要使用分為以下例子:①

12、Cocos2dx 3.0遊戲開發找小之3.0中的生命周期分析

ide () mil and 地理 splay ioe ase ima 重開發人員的勞動成果。轉載的時候請務必註明出處:http://blog.csdn.net/haomengzhu/article/details/27706303 生命周期分析 在前面文章中我

【java項目實戰】ThreadLocal封裝Connection,實現同一線程共享資源

auth @override api 數據 學習 pow 當前 程序 word 線程安全一直是程序員們關註的焦點。多線程也一直是比較讓人頭疼的話題,想必大家以前也遇到過各種各種的問題。我就不再累述了。當然,解決方案也有非常多,這篇博文給大家提供一種非常好的

Linux實戰篇:RHEL7.3 yum更換實戰

yum個人筆記分享(在線閱讀):http://note.youdao.com/noteshare?id=cdae09cf51bf77a4e94a2e2865562dbbPDF版本下載http://down.51cto.com/data/2323064本文出自 “人才雞雞” 博客,請務必保留此出處http://

python機器學習實戰

方法 baidu classes getter 全部 ken array數組 app 產生 python機器學習實戰(三) 版權聲明:本文為博主原創文章,轉載請指明轉載地址 www.cnblogs.com/fydeblog/p/7277205.html 前言 這篇博客是

Vue2+VueRouter2+webpack 構建項目實戰):配置路由,運行頁面

margin not found sans product mage -a nod targe fig 制作.vue模板文件 通過前面的兩篇博文的學習,我們已經建立好了一個項目。問題是,我們還沒有開始制作頁面。下面,我們要來做頁面了。 我們還是利用 http://cno

Spring 實戰-第章-條件化的bean

reg itl tcl span init bsp 測試 art port 在使用的時候,某些bean需要在某些特定條件化才能實例化,spring中使用的@Condition註解實現這個功能。 1.接口 package main.java.soundsystem;

實戰編程】編寫0號中斷處理程序

查看 com col 長度 獲取 pre p s 更改 end 題目:編寫0號中斷處理程序,在除法溢出時,在屏幕中間顯示字符串“hacker by admin!” 之前先補充一個rep movsb的指令知識 movsb和movsw是相反的,