Android MVP设计模式的最佳实现

相信大家看这篇文章的时候,应该都是积累了一定的Android实际项目开发经验的,大家一定都是这么经历过来的:所有的业务逻辑实现以及一些界面相关(Dialog,PopWindow....显示)还有网络请求的CallBack都放在Activity里面,从界面的初始化,数据加载,然后根据网络返回显示界面的不同内容,少则一个activity几百行代码,多则上千。我想如果稍微出点bug或者说需求做一些稍微的改动,大家就会烦恼,因为Activity所涉及的业务太多,逻辑太繁杂。可能找一个bug又得从头看一遍代码。下面介绍一下MVP设计模式,从此解耦界面与业务逻辑,轻轻松松对付产品,对付bug.


MVP设计模式简介:

其实说起MVP设计模式,应该要先去了解一下mvc设计模式,android本身就是一种MVC的设计模式:网络,数据库,本地文件的数据存取作为Model,Acitivity作为Controller,而xml文件还有一些界面相关的组件作为View。但是这样的设计在我们真正去做项目的时候可能完美的体现出来么?这是不可能的,往往我们要在Activity里面写很多界面的东西,比如Dialog,PopWindow等等,也就是说,View层和Controller层紧紧联系在一起,造成代码的高度耦合。而且在MVC里,是允许View层直接访问Model层的。而mvp的出现就是用Presenter代替以前的Controller,实现V,C层的完美分离,当然这个实现需要接口的辅助。看完下面的例子就明白了。


MVP完美实现:

1.定义View接口:

public interface ILoginView extends IBaseView {
    void showNetWorkError(); //显示网络错误

    void showLogining();//显示登录中

    void showLoginSuccess();//显示登录成功

     void showLoginFail(String msg);//显示登录失败,并说明服务器登录错误的原因
}
IBaseView其实就是一个空的interface,这个具体根据你的项目而定,看看界面需要定义的共同方法是什么,可以加到父接口中,一般用空接口就行了。

2.定义Presenter

public class LoginPresenter extends MvpBasePresenter<ILoginView> {
    public void login(Context context, String username, String password) {
        if (isViewAttached()) {//判断界面是否还在(用户没有点返回键)
            if (Utils.hasNetWork(context)) { //无网络,显示网络无连接
                getView().showNetWorkError();
                return;
            }
            getView().showLogining(); //显示正在登录中
            DataModel.login(context, username, password, new NetWorkCallBack() { //访问Model
                @Override
                public void onSuccess(String res) {
                    if (isViewAttached()) {
                        getView().showLoginSuccess(); //显示登录成功
                    }
                }

                @Override
                public void onFail(String failMsg) {
                    if (isViewAttached()) {
                        getView().showLoginFail(failMsg);//显示登录失败
                    }
                }
            });
        }
    }
}
在看一看MvpBasePresenter(可以作为模板代码直接使用)

public class MvpBasePresenter<V extends MvpView> implements MvpPresenter<V> {

    private WeakReference<V> viewRef;

    /**
     * 界面创建,Presenter与界面取得联系
     */
    @Override
    public void attachView(V view) {
        viewRef = new WeakReference<V>(view);
    }

    /**
     * 获取界面的引用
     */
    protected V getView() {
        return viewRef == null ? null : viewRef.get();
    }

    /**
     * 判断界面是否销毁
     */
    protected boolean isViewAttached() {
        return viewRef != null && viewRef.get() != null;
    }

    /**
     * 界面销毁,Presenter与界面断开联系
     */
    @Override
    public void detachView(boolean retainInstance) {
        if (viewRef != null) {
            viewRef.clear();
            viewRef = null;
        }
    }
}
3,最后定义Activity实现View接口
public class LoginActivity extends MvpActivity<ILoginView, LoginPresenter> implements ILoginView {
    private EditText etUserName, etPassword;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        findViewById(R.id.btn_login).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                presenter.login(LoginActivity.this, etUserName.getText().toString(), etPassword.getText().toString());
            }
        });
    }

    @Override
    protected LoginPresenter createPresenter() {
        return new LoginPresenter(); //具体界面对应的presenter由具体界面决定
    }

    @Override
    public void showNetWorkError() {
        Toast.makeText(this, "网络有问题,请检查网路连接", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void showLogining() {
        Toast.makeText(this, "正在登录中", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void showLoginSuccess() {
        Toast.makeText(this, "登录成功", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void showLoginFail(String msg) {
        Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
    }
}
看看MvpActivity(可以作为模板代码使用)

public abstract class MvpActivity<V extends MvpView, P extends MvpPresenter> extends Activity implements MvpView {

    protected P presenter;

    @SuppressWarnings("unchecked")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        presenter = createPresenter();
        presenter.attachView(this); //presenter取得与界面的联系
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        presenter.detachView(false);//presenter断开与界面的联系
    }

    protected abstract P createPresenter();
}

好了整个MVP的模板写完了,大家回头看一看,如果这个时候产品跟你说,我这些提示不想要Toast来提示,你改成对话框的形式,那简单啊,直接改Activity那些show方法的实现就可以了,根本不需要再查看一遍逻辑,V C分离就是好,改起来超快。

仅仅只有这一个优点么?

No,往往我们在开发产品的时候,UI和接口很难达到同步,这个时候MVP就派上用场了,如果界面出了,接口没出,我们是不是可以先把Activity和Presenter给实现了呢?等到接口出来,我们直接写Model层的代码就ok了,再也不需要傻傻的等着了。

有人说mvp会导致接口泛滥,但是你用了之后,不觉得多写两个接口就可以方便这么多,而且维护方便了不少么?




  • 6
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
### 回答1: MVP(Model-View-Presenter)是一种用于构建用户界面的设计模式,它将界面分为三个主要部分:模型(Model)、视图(View)和展示者(Presenter)。在Qt框架中实现MVP设计模式的demo可以按照以下步骤进行: 1. 首先,创建一个数据模型类(Model),该类负责处理数据的获取和管理。可以定义一些成员变量用于保存数据,并提供相应的方法来操作这些数据。 2. 接下来,创建一个视图类(View),该类用于用户界面的展示。可以在视图类中使用Qt的控件来构建界面,并将用户输入的操作反馈给Presenter类处理。 3. 然后,创建一个展示者类(Presenter),该类作为Model和View之间的中介,处理业务逻辑和数据的交互。可以在Presenter中实例化Model和View对象,并定义一些方法用于处理用户界面的操作,同时更新数据模型。 4. 最后,在Qt的主函数中初始化Model、View和Presenter对象,并建立它们之间的连接。在实际的应用场景中,可以通过信号和槽机制来实现View和Presenter之间的通信,以及Model和Presenter之间的数据传递。 通过上述步骤的实现,就可以在Qt框架中创建一个简单的MVP设计模式的demo。在这个demo中,View负责用户界面展示,Presenter处理业务逻辑和数据交互,Model负责数据的管理。这种设计模式的好处是使代码结构清晰,职责明确,易于扩展和维护,使得开发过程更加高效。 ### 回答2: MVP(Model-View-Presenter)是一种常用的设计模式,用于将业务逻辑和用户界面分离。Qt框架是一个功能强大的GUI开发框架,可以轻松地实现MVP设计模式。 首先,我们需要创建三个主要的组件:Model(模型)、View(视图)和Presenter(展示器)。 模型(Model)是应用程序的数据源。它负责处理数据的读取、写入和处理。在Qt中,我们可以使用QAbstractItemModel类或自定义的数据结构作为模型。 视图(View)是用户界面的呈现层。它负责展示数据,并且可以与用户进行交互。在Qt中,我们可以使用QWidget或QML作为视图。 展示器(Presenter)是模型和视图之间的桥梁。它负责接收视图的用户交互事件,并使用模型来处理业务逻辑。在Qt中,我们可以使用QObject类来实现展示器。 在实现MVP设计模式的Demo中,我们可以创建一个简单的待办事项列表应用程序。 首先,在模型中,我们可以定义一个代表待办事项的数据结构,并实现数据的增删改查操作。 接下来,在视图中,我们可以使用QWidget或QML来创建一个用于展示待办事项的列表的界面。用户可以添加、删除和更新待办事项。 最后,在展示器中,我们可以连接视图和模型,处理用户的交互事件,并将其反映到模型中。 例如,当用户点击添加按钮时,展示器将接收到该事件,并调用模型的添加方法。模型更新后,展示器将更新视图以反映最新的待办事项列表。 通过以上步骤,我们可以实现一个简单的MVP设计模式的Demo,同时利用Qt框架的强大功能。这样做将使代码更易于维护和扩展,并促进良好的代码分离和可测试性。 ### 回答3: MVP(Model-View-Presenter)是一种软件设计模式,主要用于分离应用程序的业务逻辑和图形界面。在Qt中实现MVP设计模式的示例可以按照以下步骤进行: 1. 创建模型(Model):模型负责处理应用程序的业务逻辑和数据处理。可以使用Qt的数据结构,如QList、QMap等来表示数据。例如,在一个学生信息管理系统中,可以创建一个学生模型类(StudentModel),用于存储和处理学生的信息。 2. 创建视图(View):视图是用户界面的可视化部分,负责读取模型中的数据并展示给用户。可以使用Qt的界面设计工具(如Qt Designer)创建视图的UI界面。例如,在学生信息管理系统中,可以创建一个学生信息展示的窗口,包含姓名、年龄、性别等信息的文本框和标签。 3. 创建Presenter(Presenter):Presenter是连接模型和视图之间的桥梁,负责处理用户的输入操作,更新模型中的数据,并将更新后的数据传递给视图进行展示。可以使用Qt的信号与槽机制来实现Presenter的交互逻辑。例如,在学生信息管理系统中,可以创建一个学生信息展示的Presenter类(StudentPresenter),用于响应用户的操作,更新学生模型的数据,并将更新后的数据传递给学生信息展示的视图进行展示。 4. 其他辅助类:除了上述三个核心类外,还可以根据需要创建其他辅助类,用于处理一些通用的功能,如数据验证、数据转换等。 在实现MVP设计模式的Demo中,可以创建一个学生信息管理系统的应用程序。通过使用上述步骤创建模型、视图和Presenter,并在主函数中进行连接和初始化,以实现学生信息的展示和更新操作。用户可以通过视图输入学生信息,并通过Presenter将输入的信息更新到模型中,并实时反映在视图中。 通过以上步骤以及合理的类和模块的划分,可以实现MVP设计模式的Demo,并帮助开发人员更好地分离业务逻辑和图形界面,提高代码的可维护性和可扩展性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值