BaseDemo

介绍

BaseDemo 是Android MVVM + Retrofit + OkHttp + Coroutine 协程 + Room + 组件化架构的Android应用开发规范化架构,通过不断的升级迭代,目前主要分为两个版本,分别为分支 MVVM+Databinding 组件化版本,分支MVVM+Databinding+Single 单体版本。旨在帮助您快速构建属于自己的APP项目架构,做到快速响应上手,另外再长期的实践经验中汇总了大量的使用工具类,主要放在了项目 lib_common组件中,以供大家参考使用。具体使用请开发者工具自己项目需求决定选择如何使用。

Github :https://github.com/zhouhuandev/BaseDemo

Gitee(国内镜像) :https://gitee.com/shandong_zhaotai_network_sd_zhaotai/BaseDemo

如果我的付出可以换来对您的帮助的话,还请您点个start,将会是我不懈更新的动力,万分感谢。如果在使用中有任何问题,请留言

  • 电子邮件:zhouhuandev@gmail.com

功能演示

软件架构说明

上拉加载,下拉刷新

演示入口

SaveStateHandler操作示例

Room操作示例

  • 扫码体验APK

  • 演示视频传送门

主要功能

核心基础库 lib_base

MVVM十大的核心公用基类和六大基础页面接口

基础页面接口
  • 视图层核接口 BaseView
interface BaseView : ILoadView, INoDataView, ITransView, INetErrView {fun initListener()fun initData()fun finishActivity()
}
  • 加载初始化弹窗接口 ILoadView
interface ILoadView {//显示初始加载的View,初始进来加载数据需要显示的Viewfun showInitLoadView()//隐藏初始加载的Viewfun hideInitLoadView()
}
  • 显示是否有数据页面接口 INoDataView
interface INoDataView {//显示无数据Viewfun showNoDataView()//隐藏无数据Viewfun hideNoDataView()//显示指定资源的无数据Viewfun showNoDataView(@DrawableRes resid: Int)
}
  • 显示小菊花View接口 ITransView
interface ITransView {//显示背景透明小菊花View,例如删除操作fun showTransLoadingView()//隐藏背景透明小菊花Viewfun hideTransLoadingView()
}
  • 显示是否网络错误View接口 INetErrView
interface INetErrView {//显示网络错误的Viewfun showNetWorkErrView()//隐藏网络错误的Viewfun hideNetWorkErrView()
}
  • 基础刷新接口 BaseRefreshView
interface BaseRefreshView {/*** 是否启用下拉刷新* @param b*/fun enableRefresh(b: Boolean)/*** 是否启用上拉加载更多*/fun enableLoadMore(b: Boolean)/*** 刷新回调* 向 ViewModel 发送刷新请求*/fun onRefreshEvent()/*** 加载更多的回调* 向 ViewModel 发送加载更多请求*/fun onLoadMoreEvent()/*** 自动加载的事件* 向 ViewModel 发送自动加载的请求*/fun onAutoLoadEvent()/*** 停止刷新*/fun stopRefresh()/*** 停止加载更多*/fun stopLoadMore()/*** 自动加载数据*/fun autoLoadData()
}
BaseActivity
abstract class BaseActivity : RxAppCompatActivity(), BaseView {abstract fun onBindLayout(): Intabstract fun initView()abstract override fun initData()override fun initListener()
}
BaseMvvmActivity
abstract class BaseMvvmActivity<VM : BaseViewModel> : BaseActivity() {/*** 绑定 ViewModel*/abstract fun onBindViewModel(): Class<VM>/*** 放置 观察者对象*/abstract fun initViewObservable()
}
BaseMvvmDataBindingActivity
abstract class BaseMvvmDataBindingActivity<V : ViewDataBinding, VM : BaseViewModel> : BaseMvvmActivity<VM>() {abstract fun onBindVariableId(): Int
}
BaseMvvmRefreshActivity
abstract class BaseMvvmRefreshActivity<T, VM : BaseRefreshViewModel<T>> : BaseMvvmActivity<VM>(), BaseRefreshView {protected abstract fun onBindRreshLayout(): Intprotected abstract fun enableRefresh(): Booleanprotected abstract fun enableLoadMore(): Boolean
}
BaseMvvmRefreshDataBindingActivity
abstract class BaseMvvmRefreshDataBindingActivity<T, V : ViewDataBinding, VM : BaseRefreshViewModel<T>> : BaseMvvmDataBindingActivity<V, VM>(), BaseRefreshView {protected abstract fun onBindRreshLayout(): Intprotected abstract fun enableRefresh(): Booleanprotected abstract fun enableLoadMore(): Boolean
}
BaseFragment
    abstract fun onBindLayout(): Intabstract fun initView(mView: View)abstract override fun initData()
BaseMvvmFragment
abstract class BaseMvvmFragment<VM : BaseViewModel> : BaseFragment() {abstract fun onBindViewModel(): Class<VM>abstract fun initViewObservable()
}
BaseMvvmDataBindingFragment
abstract class BaseMvvmDataBindingFragment<V : ViewDataBinding, VM : BaseViewModel> :BaseMvvmFragment<VM>() {abstract fun onBindVariableId(): Int
}
BaseMvvmRefreshFragment
abstract class BaseMvvmRefreshFragment<T, VM : BaseRefreshViewModel<T>> : BaseMvvmFragment<VM>(),BaseRefreshView {protected abstract fun onBindRreshLayout(): Intprotected abstract fun enableRefresh(): Booleanprotected abstract fun enableLoadMore(): Boolean
}
BaseMvvmRefreshDataBindingFragment
abstract class BaseMvvmRefreshDataBindingFragment<T, V : ViewDataBinding, VM : BaseRefreshViewModel<T>> :BaseMvvmDataBindingFragment<V, VM>(),BaseRefreshView {protected abstract fun onBindRreshLayout(): Intprotected abstract fun enableRefresh(): Booleanprotected abstract fun enableLoadMore(): Boolean
}
  • BaseAdapter
  • BaseDataBindAdapter
  • BaseViewHolder
  • IBaseViewModel
  • BaseViewModel
  • BaseRefreshViewModel

功能特色

  • 支持是否使用 ToolBar
    open fun enableToolbar(): Boolean {return true}
  • 支持自定义 ToolBar
    open fun onBindToolbarLayout(): Int {return R.layout.common_toolbar}
  • 支持标题文字图标信息自定义
    open fun getTootBarTitle(): String {return ""}/*** 设置返回按钮的图样,可以是Drawable ,也可以是ResId* 注:仅在 enableToolBarLeft 返回为 true 时候有效** @return*/open fun getToolBarLeftIcon(): Int {return R.drawable.ic_white_black_45dp}/*** 是否打开返回** @return*/open fun enableToolBarLeft(): Boolean {return false}/*** 设置标题右边显示文字** @return*/open fun getToolBarRightTxt(): String {return ""}/*** 设置标题右边显示 Icon** @return int resId 类型*/open fun getToolBarRightImg(): Int {return 0}/*** 右边文字监听回调* @return*/open fun getToolBarRightTxtClick(): View.OnClickListener? {return null}/*** 右边图标监听回调* @return*/open fun getToolBarRightImgClick(): View.OnClickListener? {return null}
  • 支持loading加载数据
    override fun showInitLoadView() {showInitLoadView(true)}override fun hideInitLoadView() {showInitLoadView(false)}
  • 支持透明loading的加载数据
    override fun showTransLoadingView() {showTransLoadingView(true)}override fun hideTransLoadingView() {showTransLoadingView(false)}
  • 支持是否使用全屏显示
    open fun enableAllowFullScreen(): Boolean {return false}
  • 支持显示无数据
    override fun showNoDataView() {showNoDataView(true)}override fun showNoDataView(resid: Int) {showNoDataView(true, resid)}override fun hideNoDataView() {showNoDataView(false)}
  • 支持网络网络错误显示
    override fun hideNetWorkErrView() {showNetWorkErrView(false)}override fun showNetWorkErrView() {showNetWorkErrView(true)}
  • 支持Fragment的懒加载
    /*** 懒加载机制 当页面可见的时候加载数据* 如果当前 FragmentTransaction.setMaxLifecycle 处理 Lifecycle.State.RESUMED 则 懒加载失效* 如果 FragmentTransaction.setMaxLifecycle 传入BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT ,* 则只有当前的Fragment处于Lifecycle.State.RESUMED状态。 所有其他片段的上限为Lifecycle.State.STARTED 。* 如果传递了BEHAVIOR_SET_USER_VISIBLE_HINT ,则所有片段都处于Lifecycle.State.RESUMED状态,* 并且将存在Fragment.setUserVisibleHint(boolean)回调*/private fun lazyLoad() {//这里进行双重标记判断,必须确保onCreateView加载完毕且页面可见,才加载数据if (isViewCreated && isViewVisable) {Log.d(TAG, "lazyLoad: Successful")initData()//数据加载完毕,恢复标记,防止重复加载isViewCreated = falseisViewVisable = false} else {Log.d(TAG, "lazyLoad: Fail")}}//默认不启用懒加载open fun enableLazyData(): Boolean {return false}
  • 支持DataBinding
  • 封装UIChangeLiveDataUIChangeRefreshLiveData
  • ViewModel lazy加载

上拉下拉功能组件 lib_refresh_layout

  • 支持最基本的下拉刷新、上拉加载更多
  • 支持自定义HeadView和FootView
  • 支持自动刷新
  • 支持启用、禁用下拉刷新
  • 支持启用、进攻上拉加载更多
  • 通用小菊花样式DaisyRefreshLayout
  • 通用小箭头样式ArrowRefreshLayout

项目架构

  • 集成模式:所有的业务组件被“app壳工程”依赖,组成一个完整的APP;
  • 组件模式:可以独立开发业务组件,每一个业务组件就是一个APP;
  • app壳工程:负责管理各个业务组件,和打包apk,没有具体的业务功能;
  • 业务组件:根据公司具体业务而独立形成一个的工程;
  • 功能组件:提供开发APP的某些基础功能,例如打印日志、下拉刷新控件等;
  • Main组件:属于业务组件,指定APP启动页面、主界面;
  • Common组件:属于功能组件,支撑业务组件的基础,提供多数业务组件需要的功能

MVVM架构

BaseMVVM架构

官方指导MVVM架构

  • View层类关系图

  • ViewModel层类关系图

组件化实现

基于阿里 ARouter 作为路由,实现组件与组件的通信跳转

https://github.com/alibaba/ARoute

集成模式和组件模式转换

Module的属性是在每个组件的 build.gradle 文件中配置的,当我们在组件模式开发时,业务组件应处于application属性,这时的业务组件就是一个 Android App,可以独立开发和调试;而当我们转换到集成模式开发时,业务组件应该处于 library 属性,这样才能被我们的“app壳工程”所依赖,组成一个具有完整功能的APP

先打开 BaseDemo 工程的根目录下找到 gradle.properties 文件,然后将 isModule 改为你需要的开发模式(true/false), 然后点击 Sync Project 按钮同步项目

isModule=false

if (isModule.toBoolean()) {apply plugin: 'com.android.application'
} else {apply plugin: 'com.android.library'
}

组件之间AndroidManifest合并问题

我们可以为组件开发模式下的业务组件再创建一个 AndroidManifest.xml,然后根据isModule指定AndroidManifest.xml的文件路径,让业务组件在集成模式和组件模式下使用不同的AndroidManifest.xml,这样表单冲突的问题就可以规避了 已module_main组件为例配置如下:

sourceSets {main {jniLibs.srcDirs = ['libs']if (isModule.toBoolean()) {manifest.srcFile 'src/main/module/AndroidManifest.xml'} else {manifest.srcFile 'src/main/AndroidManifest.xml'java {exclude 'debug/**'}}}}

组件模式下的Application

在每个组件的debug目录下创建一个Application并在module下的AndroidManifest.xml进行配置

集成开发模式下的Application

排除掉 debug/** 下面的Application,使用 moudle_app 壳子下面的 Application,其必须继承 ModuleApplication

class MyApp : ModuleApplication()

build.gradle 管理

  • 第三方 lib 源码库 且都需引入 lib.build.gradle,每个单独 moudle 都需要引入 module.build.gradle
  • versions.gradle 依赖三方库版本统一管理
  • base.build.gradle 基础编译版本统一管理

开发环境

  • Android studio:3.6、4.0
  • gradle版本:.6.5
  • gradle android plugin:4.0.2

版本规范

  • minSdkVersion(最低适配版本):21 (Android 5.0)
  • targetSdkVersion(最高适配版本):30 (Adnroid 10.0)
  • compileSdkVersion(编译版本):30 (Adnroid 10.0)
  • buildToolsVersion(SDK 构建版本):30.0.2

问题反馈

欢迎 Start ,打call https://github.com/zhouhuandev/BaseDemo 在使用中有任何问题,请留言

国内镜像:https://gitee.com/shandong_zhaotai_network_sd_zhaotai/BaseDemo

邮箱:zhouhuandev@gmail.com

关于作者

Name : "zhouhuandev",
Blog : "https://blog.csdn.net/youxun1312"
Github:"https://github.com/zhouhuandev/BaseDemo"
Gitee:"https://gitee.com/shandong_zhaotai_network_sd_zhaotai/BaseDemo"

感谢第三方开源

  • YImagePicker
  • KLog
  • SmartRefreshLayout
  • BRVAH

注:还有一大部分对此项目提供帮助的开源大佬们,就不一一列出,在此衷心感谢,若是有不足地方麻烦指导改正。

日志

  • 2021.8.14 优化整体架构,升级依赖版本
  • 2021.11.23 优化架构,更新刷新框架、Adapter适配器,修复已知问题

License

Copyright (C)  zhouhuan, BaseDemo Framework Open Source ProjectLicensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License athttp://www.apache.org/licenses/LICENSE-2.0Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
ProjectLicensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License athttp://www.apache.org/licenses/LICENSE-2.0Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

Android MVVM + Retrofit + OkHttp + Coroutine 协程 + Room + 组件化架构的Android应用开发规范化架构相关推荐

  1. Android开发之MVVM模式实践:协程与网络请求的结合

    前言 大家好,我是小益!在经过前两章对协程的介绍后,我们终于又回到了 MVVM的封装 .协程在Android开发中最常用的场景应该是网络请求了,其次是一些使用 Thread 的场景,本章内容我们将着重 ...

  2. coroutine协程详解

    前两天阿里巴巴开源了coobjc,没几天就已经2千多star了,我也看了看源码,主要关注的是协程的实现,周末折腾了两整天参照Go的前身libtask和风神的coroutine实现了一部分,也看了一些文 ...

  3. Android之网络请求通过协程+okhttp的没有做网络异常处理导致程序奔溃问题

    1 问题 app里面的网络请求是通过协程+okhttp来实现的,但是没有做网络异常处理(域名无法解析.502错误等等一系列),导致程序奔溃 2 尝试 因为app基本上做好了,外面有大几十个地方调用,然 ...

  4. 利用swoole coroutine协程实现redis异步操作

    <?php #注意:可能会遇到这样的现象,用swoole协程的方法访问常规方法添加到redis中的数据,可能访问不到(直接返回NULL)!这可能是两者采用了不同的技术标准所致!项目中要统一标准 ...

  5. Thread(线程)、Fiber(纤程)、coroutine(协程) 、绿色线程(GreenThread)

    计算机有进程,线程和协程.前两者大家都知道,很常见的玩意.而协程,则是基于线程之上的,自主开辟的异步任务,很多人更喜欢叫它们纤程(Fiber),或者绿色线程(GreenThread). 进程 为了使多 ...

  6. 学会Retrofit+OkHttp+RxAndroid三剑客的使用,让自己紧跟Android潮流的步伐

    概括 在上一篇博客android网络框架OkHttp之get请求(源码初识) 讲解了OkHttp的简单使用和OkHttp源码的分析,主要讲解的还是理论上的知识,还是没有去实践下,那么这篇博客里面就来实 ...

  7. XLua Coroutine协程

    1.将XLua的util文件 放在自己的项目当中,在require 它 XLua中的位置 移动到你项目的自定义的位置 2.在Lua层 封装XLua的 Corrutine 功能,代码如下 corouti ...

  8. Lua学习——Coroutine协程

    在菜鸟教程学习lua时,在coroutine这一章看到这么一段 coroutine.running就可以看出来,coroutine在底层实现就是一个线程. 当create一个coroutine的时候就 ...

  9. Go协程通道锁故事化理解

    通道的故事 老大交给小兵任务,从仓库运大米到老大家,小兵就去仓库取到大米,找到大米之后,运到老大家,如果不运到老大家的话,就会一直扛着,身体受不了,就会dead lock.必须有老大家进行接收. 老大 ...

最新文章

  1. 【每日DP】day4 P1417 烹调方案(奇怪的01背包增加了)难度⭐⭐⭐
  2. dsst依赖opencv
  3. Python学习笔记:进程和线程(承)
  4. java流与文件——文本输入输出
  5. mysql集群跨地域同步部署_跨地域冗余 - 跨数据中心部署方案 - 《TiDB v2.1 用户文档》 - 书栈网 · BookStack...
  6. PHP配置,php.ini以及覆盖问题
  7. oracle运行环境配置,oracle 环境配置开机自启动
  8. java实现手机开关机_Android 系统重启与关机:Java 代码实现
  9. python json串中文乱码问题
  10. 【BZOJ1060】[ZJOI2007] 时态同步(树形DP)
  11. 判断某一点是否在三角形内
  12. C++11 std::ref()
  13. 很牛的vim使用技巧nbsp;转载来的
  14. 数学中几种积:点积(数量积/标量积/内积)、叉积(叉乘/向量积)、外积(张量积/Kronecker积)、哈达玛积(元素积)
  15. 安装Linux启动盘【准备8G以上U盘即可】
  16. 老师讲,这是个hin重要的C语言习题
  17. 知识改变命运 学习成就梦想
  18. Python可视化模块——Matplotlib(2)
  19. APT32F102-SIO模块控制WS2812
  20. 一种使用Python自实现KMeans++聚类算法的写法

热门文章

  1. Spring IOC笔记——DI(依赖注入)的使用详解
  2. 弹性波波长计算公式_什么是弹性波的相干波长
  3. iphone 数据导android,一招解决iPhone数据导入vivo手机,网友直说,太方便
  4. 世界十大蓝牙耳机品牌,四款好用的蓝牙耳机推荐
  5. erlang的Emakefile和批处理启动
  6. 从键盘任意输入一个实数,不使用计算绝对值函数编程计算并输出该实数的绝对值。
  7. 网络中的一些基本概念
  8. 鄂尔多斯计算机考试系统,鄂尔多斯驾驶员理论模拟考试系统
  9. 真二次元!动漫形象风格迁移
  10. 传奇gm命令怎么用_热血传奇1.76gm口令大全介绍