Android 快速打造Kotlin+MVP+Rxjava+Retrofit項目架構(gòu)(引用Google推薦架構(gòu)模式:契約接口)

快速打造Kotlin+MVP+Rxjava+Retrofit項目架構(gòu)(引用Google推薦架構(gòu)模式:契約接口)

文件目錄:

主目錄

mvp中的目錄

主要代碼示例:

  • 1溯泣、一個登錄模型(也叫bean、entity)
/**
 * 后臺規(guī)定的接口命名的字段是大寫開頭
 */

class LoginModel : Base() {

    /**
     * Ext : {"Authority":3,"Token":"213bfc688a7e099b1f2a27b64f59c9e1"}
     * Message : 登錄成功
     * State : true
     */

    var ext: ExtBean? = null


    class ExtBean {
        /**
         * Authority : 3
         * Token : 213bfc688a7e099b1f2a27b64f59c9e1
         */

        var authority: Int = 0
        var token: String? = null
    }
}
  • 2、一個登錄的契約接口
interface View : BaseContract.BaseView {
        fun loginSus(loginModel: LoginModel) //登錄成功
        fun err(code: Int, message: String) //出錯
    }

    interface Presenter : BaseContract.BasePresenter<View> {
        fun login(telephoneNumber: String, verificationCode: String, token: String) //登錄
    }

順便貼出BaseContract契約接口基類

/**
 * 對應(yīng)mvp中的contract契約接口
 */

interface BaseContract {
    interface BasePresenter<in T> {
        /*該方法可以獲取到View實例對象*/
        fun attachView(view: T)

        /*釋放View對象的引用锈津,gc才能回收View*/
        fun detachView()
    }

    interface BaseView {
        //view的一些共有方法
        fun showError(e: Throwable)
        fun complete()
    }
}
  • 3暮顺、一個實現(xiàn)了View接口充當(dāng)V層的Activity
/**
 * 模擬登錄的view層
 */
class MainActivity : BaseActivity<LoginContract.View,LoginContract.Presenter>(),LoginContract.View {
    override fun err(code: Int, message: String) {
        ToastUtils.showShort("錯誤碼:$code\n錯誤信息:$message")
    }

    override fun loginSus(loginModel: LoginModel) {
        disMissLoading()
        if (loginModel.State) {
            ToastUtils.showShort("登錄成功")
        }else{
            ToastUtils.showShort(loginModel.Message)
        }
    }

    override fun showError(e: Throwable) {
        disMissLoading()
    }

    override fun complete() {

    }

    override fun getLayoutId(): Int = R.layout.activity_main

    override fun configView() {
        login.setOnClickListener({
            showLoading()
            mPresenter!!.login("18381309101","1111","") })
    }

    override fun initData() {

    }

    override fun initPresenter(): LoginContract.Presenter = LoginPresenter()

}
  • 實現(xiàn)Presenter,充當(dāng)P層的一個登錄的Presenter
class LoginPresenter : RxPresenter<LoginContract.View>(), LoginContract.Presenter {

    /**
     * 登錄
     * @param telephoneNumber
     * @param verificationCode
     * @param token
     */
    override fun login(telephoneNumber: String, verificationCode: String, token: String) {
        val subscription = HttpManager.getWorkHttpService()!!.signIn(telephoneNumber, verificationCode, token)
                .compose(ScheduleTransformer.instance)
                .subscribe(object : ObserverImp<Any>() {

                    override fun onErr(errCode: Int, str: String) {
                        mView!!.err(errCode,str)
                    }

                    override fun doNext(loginModel: Any) {
                        mView!!.loginSus(loginModel as LoginModel)
                    }
                })
        addSubscribe(subscription)
    }
}

到這里mvp構(gòu)建就已經(jīng)完成了响驴,但是那些日常操作:統(tǒng)一錯誤處理透且、文件進(jìn)度監(jiān)聽、Retorfit的自定義攔截器豁鲤、內(nèi)存泄漏處理秽誊、Retrofit自定義轉(zhuǎn)換器等等還需要完成。琳骡。锅论。

一些重要的代碼:

  • 統(tǒng)一錯誤處理
override fun onError(e: Throwable) {
        var e = e
        var throwable = e
        //獲取最根源的異常
        while (throwable.cause != null) {
            if (e is HttpException) {
                break
            }
            e = throwable
            throwable = throwable.cause!!
        }
        if (e is HttpException) {
            when (e.code()) {
                UNAUTHORIZED -> onErr(UNAUTHORIZED, "")
                FORBIDDEN -> onErr(FORBIDDEN, "權(quán)限錯誤")          //權(quán)限錯誤,需要實現(xiàn)
                NOT_FOUND -> onErr(NOT_FOUND, "")
                REQUEST_TIMEOUT -> onErr(REQUEST_TIMEOUT, "")
                GATEWAY_TIMEOUT -> onErr(GATEWAY_TIMEOUT, "")
                INTERNAL_SERVER_ERROR -> onErr(INTERNAL_SERVER_ERROR, "")
                BAD_GATEWAY -> onErr(BAD_GATEWAY, "")
                SERVICE_UNAVAILABLE -> onErr(SERVICE_UNAVAILABLE, "")
                else -> onErr(ERR_CODE_NET, "")
            }
        } else if (e is SocketTimeoutException) {
            onErr(GATEWAY_TIMEOUT, "請求超時!")
        } else if (e is UnknownHostException) {
            onErr(ERR_CODE_NET, "網(wǎng)絡(luò)連接失敗!")
        } else {
            onErr(ERR_CODE_UNKNOWN, "未知錯誤!")
        }
    }
  • 2楣号、文件進(jìn)度監(jiān)聽
 override fun read(sink: Buffer, byteCount: Long): Long {
                try {
                    val bytesRead = super.read(sink, byteCount)
                    if (totalBytesRead == 0L && listener != null) {
                        listener.DLoadStart()
                    }
                    totalBytesRead += if (bytesRead != -1L) bytesRead else 0

                    if (null != listener) {
                        if (bytesRead == -1L) {
                            listener.DLoadSuccess()
                        }
                        if ((totalBytesRead * 100L / responseBody.contentLength()).toInt() > lastPro) {
                            listener.DLoadProgress((totalBytesRead * 100L / responseBody.contentLength()).toInt())
                        }
                        lastPro = (totalBytesRead * 100L / responseBody.contentLength()).toInt()
                    }
                    return bytesRead
                } catch (e: IOException) {
                    listener?.DLoadFail()
                }

                return 0L

            }
  • 3最易、自定義文件下載攔截器
/**
 * 文件下載攔截器
 */

class DownLoadInterceptor(private val listener: DLProListener) : Interceptor {

    @Throws(IOException::class)
    override fun intercept(chain: Interceptor.Chain): Response {
        val originalResponse = chain.proceed(chain.request())

        return originalResponse.newBuilder()
                .body(FileResBody(originalResponse.body(), listener))
                .build()
    }
}
  • 4、內(nèi)存泄漏處理竖席,釋放對activity 的引用
 override fun attachView(view: T) {
        this.mView = view
    }

    override fun detachView() {
        this.mView = null
        unSubscribe()
    }
  • 5耘纱、自定義單例的Retrofit轉(zhuǎn)換器
/**
 * 單例實現(xiàn)的轉(zhuǎn)換器,統(tǒng)一指定每一個接口請求實在io線程毕荐,回調(diào)在ui線程
 */

class ScheduleTransformer<T>
/**
 * 私有的構(gòu)造函數(shù)
 */
private constructor() : Observable.Transformer<T, T> {
    /**
     *懶加載內(nèi)部單例
     */
    private object TransformerHolder {
        private val instance: ScheduleTransformer<Any>? = null
        fun getInstance(): ScheduleTransformer<Any> {
            return instance ?: ScheduleTransformer<Any>()
        }
    }

    override fun call(tObservable: Observable<T>): Observable<T> {
        return tObservable.subscribeOn(Schedulers.io())
                .unsubscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
    }

    /**
     * 伴生
     */
    companion object {
        val instance: ScheduleTransformer<Any>
            get() = TransformerHolder.getInstance()
    }
}

該架構(gòu)用起來十分方便和靈活束析,里面還有常用的一些基類和處理類,已經(jīng)在正式項目中使用憎亚,請大膽嘗試员寇,Github地址:實戰(zhàn)MVP架構(gòu),包含JAVA版本第美,歡迎打星鼓勵蝶锋!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市什往,隨后出現(xiàn)的幾起案子扳缕,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,692評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件躯舔,死亡現(xiàn)場離奇詭異驴剔,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)粥庄,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,482評論 3 392
  • 文/潘曉璐 我一進(jìn)店門丧失,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人惜互,你說我怎么就攤上這事布讹。” “怎么了训堆?”我有些...
    開封第一講書人閱讀 162,995評論 0 353
  • 文/不壞的土叔 我叫張陵描验,是天一觀的道長。 經(jīng)常有香客問我蔫慧,道長挠乳,這世上最難降的妖魔是什么权薯? 我笑而不...
    開封第一講書人閱讀 58,223評論 1 292
  • 正文 為了忘掉前任姑躲,我火速辦了婚禮,結(jié)果婚禮上盟蚣,老公的妹妹穿的比我還像新娘黍析。我一直安慰自己,他們只是感情好屎开,可當(dāng)我...
    茶點故事閱讀 67,245評論 6 388
  • 文/花漫 我一把揭開白布阐枣。 她就那樣靜靜地躺著,像睡著了一般奄抽。 火紅的嫁衣襯著肌膚如雪蔼两。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,208評論 1 299
  • 那天逞度,我揣著相機(jī)與錄音额划,去河邊找鬼。 笑死档泽,一個胖子當(dāng)著我的面吹牛俊戳,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播馆匿,決...
    沈念sama閱讀 40,091評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼抑胎,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了渐北?” 一聲冷哼從身側(cè)響起阿逃,我...
    開封第一講書人閱讀 38,929評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后恃锉,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體羽历,經(jīng)...
    沈念sama閱讀 45,346評論 1 311
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,570評論 2 333
  • 正文 我和宋清朗相戀三年淡喜,在試婚紗的時候發(fā)現(xiàn)自己被綠了秕磷。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,739評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡炼团,死狀恐怖澎嚣,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情瘟芝,我是刑警寧澤易桃,帶...
    沈念sama閱讀 35,437評論 5 344
  • 正文 年R本政府宣布,位于F島的核電站锌俱,受9級特大地震影響晤郑,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜贸宏,卻給世界環(huán)境...
    茶點故事閱讀 41,037評論 3 326
  • 文/蒙蒙 一造寝、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧吭练,春花似錦诫龙、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,677評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至分尸,卻和暖如春锦聊,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背箩绍。 一陣腳步聲響...
    開封第一講書人閱讀 32,833評論 1 269
  • 我被黑心中介騙來泰國打工孔庭, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人伶选。 一個月前我還...
    沈念sama閱讀 47,760評論 2 369
  • 正文 我出身青樓史飞,卻偏偏與公主長得像,于是被迫代替她去往敵國和親仰税。 傳聞我的和親對象是個殘疾皇子构资,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,647評論 2 354

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,081評論 25 707
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)陨簇,斷路器吐绵,智...
    卡卡羅2017閱讀 134,652評論 18 139
  • 根據(jù)小碼哥擴(kuò)展班所做的筆記迹淌。 創(chuàng)建framework工程: 創(chuàng)建Tool類,編譯己单,show in finder唉窃,發(fā)...
    健了個平_24閱讀 209評論 0 0
  • 今天是9月29日,星期五纹笼。今天纹份,我們學(xué)習(xí)了拆裝發(fā)動機(jī),這個操作看起來不是很難廷痘,但是當(dāng)自己去做的時候蔓涧,就會感覺有一些...
    WMY王明陽閱讀 225評論 0 1
  • 我們曾在愛情中元暴,經(jīng)歷憂傷,喜悅兄猩。我們渴望愛茉盏,又恐懼愛。我們以為要愛一輩子的人枢冤,最后卻淪為最熟悉的陌生人鸠姨。是什...
    余文寫詩閱讀 435評論 0 1