說說我自己常用的 Android 架構

先列出一些常用的依賴,想必看到下面的依賴大家也能明白接下來要講的是什么?

//所需依賴
compile 'com.squareup.retrofit2:retrofit:2.1.0'
compile 'io.reactivex.rxjava2:rxjava:2.0.5'
compile 'io.reactivex.rxjava2:rxandroid:2.0.1'
compile 'com.squareup.retrofit2:converter-gson:2.1.0'
compile 'com.google.code.gson:gson:2.8.0'
compile 'com.jakewharton.retrofit:retrofit2-rxjava2-adapter:1.0.0'

沒錯,就是老生常談的 MVP 模式。

retrofit2作為網絡請求,gson作為json解析器。注意這里是用的最新的 rxjava2 和 jakewharton 大大開源的 retrofit2-rxjava2-adapter 作為橋接器。最新的版本可以前往各自的 Github 上查看。

mvp

基礎架構

我個人比較喜歡在項目下新建一個 Android library 的模塊,取名為core,主要作用是負責網絡層和數據層。像數據實體類,數據庫操作,SharedPreferences緩存,網絡請求都可以放在 core 模塊下,主要目的就是徹底將UI和數據層完全分開(物理層面上)。

網絡模塊

需要一個 RetrofitHelper 單例模塊支持,主要是為 OkHttp 設置請求參數屬性和初始化 Api 接口服務。

OkHttp的參數設置

OkHttp上可以設置的參數很多,像緩存,頭部信息,超時時間,重連等信息都可以在 OkHttpClient 初始化設置時統一設置。

private void initOkHttp() {
    OkHttpClient.Builder builder = new OkHttpClient.Builder();
    
    //設置統一的請求頭部參數
    builder.addInterceptor(apikey);
    //設置緩存
    builder.addNetworkInterceptor(cacheInterceptor);
    builder.addInterceptor(cacheInterceptor);
    builder.cache(cache);
    //設置超時
    builder.connectTimeout(10, TimeUnit.SECONDS);
    builder.readTimeout(20, TimeUnit.SECONDS);
    builder.writeTimeout(20, TimeUnit.SECONDS);
    //錯誤重連
    builder.retryOnConnectionFailure(true);
    okHttpClient = builder.build();
}

初始化項目 api 接口

一般來說,一個項目的網絡返回數據都有統一的返回數據,比如有一個定義好的返回碼 resultCode,數據返回信息 resultInfo,以及最重要的數據對象 returnObject。所以我們需要一個類來進行網絡數據套接。

public class ApiResponse<T> {
    private int resultCode;
    private T returnObject;
    private Object ruturnInfo;

    // get 和 set 方法
    //...
}

定義好數據類型,就輪到 Retrofit 與網絡接口進行聯動,首先需要一個能夠定義 api 接口的地方 Apis

public interface Apis {

    /**
     * 獲取啟動頁圖片
     *
     * @return
     */
    @FormUrlEncoded
    @POST("getSplashImg")
    Observable<ApiResponse<SplashImageBean>> getStartImg(@Field("uid") String uid,@Field("size") String size);

    //其他的api
    ...
}

之后回到 RetrofitHelper 中初始化接口服務。

// 接口服務
apis = getApiService(HttpUtils.BASEURL, Apis.class);

// 接口服務初始化方法
private <T> T getApiService(String baseUrl, Class<T> clz) {
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(baseUrl)
            .client(okHttpClient)
            .addConverterFactory(GsonConverterFactory.create())
            .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
            .build();
    return retrofit.create(clz);
}

搭建 MVP 架構

此方法是從 Google 的開源項目android-architecture 上的 todo?mvp?rxjava 篇演變而來。一樣需要定義 View 和 Presenter 接口以及Presenter的實現,只不過加了一層 RxPresenter 對Rxjava的優化,防止內存的泄露。

MVP的基礎類

首先要為mvp模式定下基礎接口 BaseViewBasePresenter。
Presenter 需要綁定 View 才能回調 View 里面的各種方法,所以直接在類聲明的時候將 View 綁定。
同理,View 里面需要一個 Presenter 去處理數據,故定義一個 setPresenter() 方法來提醒(所以不設置也行)。

public interface BaseView {
    void setPresenter();
}

public interface BasePresenter<T extends BaseView> {
    void attachView(T view);
    void detachView();
}

Rxjava 在 MVP 上的優化

之前在寫 MoeMusic開源項目 的時候完全沒有考慮到 Rxjava 在與 Retrofit 結合請求網絡請求的時候會存在內存泄露的問題,所以在這個模塊上利用 Rxjava 的訂閱和取消訂閱功能消除內存泄露的問題。

/**
 * @author cpacm
 * @date 2017/2/26
 * @desciption 可取消訂閱的 rxpresenter,防止rxjava引起的內存泄露
 */

public abstract class RxPresenter<T extends BaseView> implements BasePresenter<T> {
    protected T view;
    protected CompositeDisposable compositeDisposable;

    protected void unDisposable() {
        if (compositeDisposable != null) {
            compositeDisposable.clear();
        }
    }

    protected void addDisposable(Disposable disposable) {
        if (compositeDisposable == null) {
            compositeDisposable = new CompositeDisposable();
        }
        compositeDisposable.add(disposable);
    }

    @Override
    public void attachView(T view) {
        this.view = view;
    }

    @Override
    public void detachView() {
        this.view = null;
        unDisposable();
    }
}

原理很簡單,就是使用 CompositeDisposable 來訂閱 rxjava 發射的事件,之后在 detachView() 解綁的時候取消訂閱。

MVP 的簡單使用

在使用前,我們先建一個 BaseActivity 作為所有的Activity的基類,并將其生命周期與 MVP 模塊關聯起來。

public abstract class BaseActivity<T extends BasePresenter> extends AppCompatActivity implements BaseView {

    protected T presenter;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setPresenter();
        if (presenter != null) {
            presenter.attachView(this);
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (presenter != null) {
            presenter.detachView();
        }
    }
}

使用示例

需求:一個 app 的啟動頁,啟動頁圖片來自服務器。

SplashContract 中定義各個接口要回調的方法。

public interface SplashContract {

    interface View extends BaseView {
        void showSplash(SplashImageBean bean);
    }

    interface Presenter extends BasePresenter<View> {
        void getSplashData();
    }
}

SplashPresenter 中實現網絡的請求和view的回調

public class SplashPresenter extends RxPresenter<SplashContract.View> implements SplashContract.Presenter {

    private Apis zqswApis;

    public SplashPresenter() {
        zqswApis = RetrofitHelper.getInstance().getApis();
    }

    @Override
    public void getSplashData() {
        Disposable disposable = zqswApis.getStartImg("","")
                .observeOn(AndroidSchedulers.mainThread())
                .subscribeOn(Schedulers.io())
                .subscribe(new Consumer<ApiResponse<SplashImageBean>>() {
                    @Override
                    public void accept(ApiResponse<SplashImageBean> splashImageBean) throws Exception {
                        view.showSplash(splashImageBean.getReturnObject());
                    }
                }, new Consumer<Throwable>() {
                    @Override
                    public void accept(Throwable throwable) throws Exception {
                        Log.e("cpacm", throwable.toString());
                    }
                });
        addDisposable(disposable);
    }
}

SplashActivity 實現UI完成整個需求

/**
 * @author cpacm
 * @date 2017/2/16
 * @desciption 啟動界面
 */

public class SplashActivity extends BaseActivity<SplashPresenter> implements SplashContract.View {

    private ImageView bgView;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_splash);
        bgView = (ImageView) findViewById(R.id.background);
        presenter.getSplashData();
    }

    @Override
    public void setPresenter() {
        presenter = new SplashPresenter();
    }

    @Override
    public void showSplash(SplashImageBean bean) {
        Glide.with(this)
                .load(bean.getImageUrl())
                .into(bgView);
        Toast.makeText(this, bean.toString(), Toast.LENGTH_SHORT).show();
    }

}

項目源碼見下方鏈接

結論

整個框架小巧而精致,而且看起來也不怎么復雜,個人項目或者小型的團隊項目應付起來應該是綽綽有余了,不過大型的項目應該還需要擴展或者采用其他的架構來應付繁瑣的需求。


MvpDemo源碼

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,923評論 6 535
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,740評論 3 420
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 176,856評論 0 380
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,175評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,931評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,321評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,383評論 3 443
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,533評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,082評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,891評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,067評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,618評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,319評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,732評論 0 27
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,987評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,794評論 3 394
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,076評論 2 375

推薦閱讀更多精彩內容