前几天发现,在Android项目代码里有一个Activity类行数居然有1000多行,而600行左右都是逻辑控制,真正和页面控件处理相关的代码不多,虽然可以用#region <>...#endregion块包起来,但是整体来说,页面和逻辑处理揉得太紧密了,有时代码复用起来也不方便,于是,决定重构,找了一下,有MVP(Model-View-Presenter,Model层负责数据管理,View层负责页面控件数据展示与设置,Presenter负责逻辑处理,控制View层如何显示与展示数据,这种层次设计,虽然代码文件多了,但是整体逻辑划分很清晰,对于团队分工和测试很方便)风格的架构还不错,了解了一下,不难,弄了半天,虽然多了几个代码文件和函数,但是原来那个Activity类行数缩减为400多行,逻辑看起来清爽多了。
现在新建一个测试工程,来说明一下MVP是怎么使用的:
1. 新建测试工程项目:
2. 修改界面文件为如下:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" android:paddingBottom="@dimen/activity_vertical_margin" android:orientation="vertical" tools:context="cn.linjk.testmvp.MainActivity"> <TextView android:id="@+id/tv_info" android:layout_gravity="center_horizontal" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="Hello World!" android:textAlignment="center"/> <Button android:id="@+id/btn_edit_info" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_marginTop="10dp" android:text="Modify Info"/> </LinearLayout>
3. 增加界面修改接口:
package cn.linjk.testmvp.views; /** * Created by LinJK on 24/12/2016. */ public interface IMainView { String getTextViewInfo(); void setTextViewInfo(String info); }
4. 在MainActivity实现这个界面修改接口IMainView:
package cn.linjk.testmvp; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.widget.Button; import android.widget.TextView; import cn.linjk.testmvp.views.IMainView; public class MainActivity extends AppCompatActivity implements IMainView{ private TextView tvIninfo; private Button btnModifyInfo; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initViewComponents(); } @Override public String getTextViewInfo() { return tvIninfo.getText().toString(); } @Override public void setTextViewInfo(String info) { tvIninfo.setText(info); } private void initViewComponents() { tvIninfo = (TextView) findViewById(R.id.tv_info); btnModifyInfo = (Button) findViewById(R.id.btn_edit_info); } }
5.增加MainActivity控制器接口IMainViewPresenter,用于控制界面的显示与内容设置逻辑:
package cn.linjk.testmvp.presenters; /** * Created by LinJK on 24/12/2016. */ public interface IMainViewPresenter { void modifyTextViewInfo(String isssnfo); }
6.实现控制器接口:
package cn.linjk.testmvp.presenters; import cn.linjk.testmvp.views.IMainView; /** * Created by LinJK on 24/12/2016. */ public class MainViewPresenter implements IMainViewPresenter{ private IMainView iMainView; public MainViewPresenter(IMainView pIMainView) { this.iMainView = pIMainView; } @Override public void modifyTextViewInfo(String info) { iMainView.setTextViewInfo(info); } }
7. 在MainActivity增加控制器,增加控制逻辑,修改MainActivity后代码如下:
package cn.linjk.testmvp; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.TextView; import cn.linjk.testmvp.presenters.MainViewPresenter; import cn.linjk.testmvp.views.IMainView; public class MainActivity extends AppCompatActivity implements IMainView{ private TextView tvIninfo; private Button btnModifyInfo; private MainViewPresenter mainViewPresenter; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initViewComponents(); initGlobalVars(); } @Override public String getTextViewInfo() { return tvIninfo.getText().toString(); } @Override public void setTextViewInfo(String info) { tvIninfo.setText(info); } private void initViewComponents() { tvIninfo = (TextView) findViewById(R.id.tv_info); btnModifyInfo = (Button) findViewById(R.id.btn_edit_info); btnModifyInfo.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { mainViewPresenter.modifyTextViewInfo("Android MVP"); } }); } private void initGlobalVars() { mainViewPresenter = new MainViewPresenter(this); } }
8. 此时运行代码,完成目的。
从上面代码逻辑可以看到,用这种方式编写代码,MainActivity类逻辑很清晰,不存在太多的逻辑控制代码,逻辑控制都在控制器处理了,这样在团队分工也容易分配,只要把业务接口写好了,就可以把接口分配人员去实现,当然,也有一个缺点是这样在项目复杂后很容易达到65535这个方法数限制,这样另外分DEX处理就行了。这种编程模式值得推荐使用。