目前因为Controller(Activity)和View,Model没有实现解耦,造成Activity既要负责处理业务逻辑,又要负责UI显示,数据绑定,所以造成Activity里代码过多,逻辑不清楚,对代码的可读性,可维护性不是很好,所以采用一些分层模式,可以更为有效的对业务逻辑层,UI显示层,数据层进行拆分,集中分层模式有:
**甘特图
**流程图
话不多说,直接上图
1.MVC架构图
架构类型 | 构造子 |
---|---|
MVP | Model, Presenter, View |
MVC | Model, Controller ,View |
MVVM | Model, View ,ViewModel |
3.简要分析区别
话不多说,直接上图
看上图Model和View是不会发生关系的,ViewModel是把View和Model关联起来的加工厂:
MVVM优势总结:
View
和Model
双向绑定,一方的改变都会影响另一方,开发者不用再去手动修改UI的数据。额,互相自动的。findViewById
也不需要butterknife
,不需要拿到具体的View去设置数据绑定监听器等等,这些都可以用DataBinding
完成。是不是很舒服?View
和Model
的双向绑定是支持生命周期检测的,不会担心页面销毁了还有回调发生,这个由lifeCycle
完成。MVC
一样导致Activity
中代码量巨大,也不会像MVP
一样出现大量的View
和Presenter
接口。项目结构更加低耦合。2.目录结构:
lib_opensource
:第三方build.gradle
依赖,本项目主要有support、lifecycle、room、fresco、retrofit、okhttp、RxJava、ARouter
这些。lib_coremodel
: 存放MVVM
中的Model
和ViewModel
两个模块,就是数据的处理和数据与UI页面的绑定。依赖lib_opensource
库。lib_common
: 公共库,主要有各种base,各种ui组件,自定义组件,公用的Activity
、公用的Fragment
,和公用的utils
等等。依赖lib_coremodel
库。module_girls
: 子功能模块,可以在library
和application
之间切换,自己可以是一个app也可以成为别的app的一个组件模块。组件化编译时为app,反之为module
。module_news
: 新闻功能模块,可以在library
和application
之间切换,自己可以是一个app也可以成为别的app的一个组件模块。组件化编译时为app
,反之为module
app_universal
: 定制版本的app,组件化编译时module_girls
和module_news
为app,所以不能把这两个作为module加进来编译,所以组件化编译时app_universal
要依赖lib_common
库,反之就可以把 module_girls
和module_news
作为module加进来编译。app_specific
: 定制版本的app
,组件化编译时module_girls
和module_news
为app,所以不能把这两个作为module加进来编译,所以组件化编译时app_specific
要依赖lib_common
库,反之就可以把 module_girls
和module_news
作为module
加进来编译Android平台上有一些比较好的MVVM框架,其中用的比较多的是RoboBinding,Robobinding。
为了精简框架,RoboBinding移除了大量不必要的代码,比如addXXListener(),findViewById()等。
可以将难以测试的Android代码转换为普通的JUnit测试。
提供对象类型Cursor来替换 - 关系类型Cursor,因为我们已经习惯于操作对象 。
可以很容易的为任何自定义组件,第三方组件或Android widget编写属性绑定实现,简化代码,使项目易于维护
-----------------------------------------Robobiding 简单使用示例:----------------------------------------------
View层(对应android xml文件)
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:bind="http://robobinding.org/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="org.robobinding.androidmvvm.MainActivity"
tools:ignore="MissingPrefix">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
bind:text="{hello}"/> //单向绑定,修改Model属性时,会自动反映到视图中(需要实体中提供相应的getHello(),setHello()方法)。
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Name:"/>
<EditText
android:layout_width="fill_parent"
android:layout_height="wrap_content"
bind:text="${name}"/> //双向绑定,修改model属性时,会自动反映到视图中;反过来,修改视图内容,也会自动更改Model的相关属性。
</LinearLayout>
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Say Hello"
bind:onClick="sayHello"/>
</LinearLayout>
public class PresentationModel implements HasPresentationModelChangeSupport {
private PresentationModelChangeSupport changeSupport;
private String name;
public PresentationModel() {
changeSupport = new PresentationModelChangeSupport(this);
}
public String getHello() {
return name + ": hello Android MVVM(Presentation Model)!";
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name; Log.d("model", "setName(),name: " + name);
}
public void sayHello(){
changeSupport.firePropertyChange("hello");
}
@Override
public PresentationModelChangeSupport getPresentationModelChangeSupport() {
return changeSupport;
}
}
**Controller层: **
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
PresentationModel presentationModel = new PresentationModel();
ViewBinder viewBinder = createViewBinder();
View rootView = viewBinder.inflateAndBind(R.layout.activity_main, presentationModel);//将model和view进行绑定
setContentView(rootView);
}
private ViewBinder createViewBinder() {
BinderFactory reusableBinderFactory = new BinderFactoryBuilder().build();
return reusableBinderFactory.createViewBinder(this);
}
}
----------------------------------------------Robobiding 简单使用结束:------------------------------------------------
android官方支持的databinding框架使用
使用条件:
使用步骤:
dependencies { classpath "com.android.databinding:dataBinder:1.0-rc1"}
apply plugin: 'com.android.databinding'
注意
在项目编译过程中,会出现诸如以下错误:
错误一:
Error:Application and test application id cannot be the same: both are 'com.fengjr.mobile' for qihuDebugAndroidTest
此处要求test的application id和项目id不能一致,需将test 的id(即testApplicationId “com.fengjr.mobile"修改为"com.fengjr.mobile.test”)
示例
view 文件 activity_main.xml:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
<data>
<variable
name="stu"
type="com.fengjr.mobile.databind.Student" /> //指明该view文件要绑定的数据模型
</data>
<LinearLayout
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{stu.name}"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{stu.addr}"/>
</LinearLayout>
</layout>
** Model文件:**
package com.fengjr.mobile.databind;
/** * Created by gaoge on 15/9/22. */
public class Student {
private final String name;
private final String addr;
public Student(String name, String addr) {
this.name = name; this.addr = addr;
}
public String getName() {
return name;
}
public String getAddr() {return this.addr;
}
}
** 绑定过程:TestDataBindingActivity **
public class TestDataBindingAct extends Activity{
@Override
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
ActivityMainBinding binding=DataBindingUtil.setContentView(this,R.layout.activity_main);
binding.setStu(new Student("姓名", "地址")); //将Model和view(xml文件进行绑定)
}
}
在mobile module 的build.gradle文件中声明 apply "com.android.databinding"后,android编译系统会自动根据xml文件名称"activity_main"生成一个ActivityMainBinding的class 文件(生成文件的命名格式和xml一一对应)
但是项目中目前使用到google annotation注释框架,且apt的版本是1.4,会和databinding编译过程冲突,造成databinding不能自动生成ActivityBinding文件,需要在mobile module的build.gradle文件中,将classpath 'com.neenbedankt.gradle.plugins:android-apt:1.4'
改为classpath 'com.neenbedankt.gradle.plugins:android-apt:1.7'
更改Model后自动更新UI,可以通过两种方法实现:
@Bindable
注解,在set()方法中增加notifyPropertyChanged()
属性,示例:ObservableField ,ObservableBoolean ,ObservableByte ,ObservableChar ,ObservableShort ,ObservableInt ,ObservableLong,ObservableArrayMap,ObservableArrayList
)public class People {
public ObservableField<String> name = new ObservableField<>();
public ObservableInt age = new ObservableInt();
public ObservableBoolean isMan = new ObservableBoolean();
}
作者:<>曾维和
原文:原文链接