繁体   English   中英

Android MVP 从 Presenter 打开 Activity,反模式?

[英]Android MVP open Activity from Presenter, anti-pattern?

如果我从 Presenter 层打开一个Activity它会是一个反模式吗?

如果是这样,我应该从视图层管理应用程序的导航吗?

是的,这是一个反mvp模式。 基于MVP中的被动视图 ,您失去了可测试性,因为您不必在演示者中处理Android框架。

因此,最好从视图层管理应用程序的导航。

class MyPresenter {
    MyPresenter.View view;

    void backButtonClicked() {
        view.navigateToHomeScreen();
    }

    public interface View {
        void navigateToHomeScreen();
    }
}

class MyActivity extends Activity implements MyPresenter.View {
    @Override
    void navigateToHomeScreen() {
        startActivity(...)
    }

    @OnClick(R.id.my_button)
    void onClick() {
        presenter.backButtonClicked();
    }
} 

这种方式的另一个优点是可以很容易地用片段或视图替换活动。

编辑1:

Morgwai说这种方式会打破关注和单一责任的分离,但你不可能在每个地方都有单一的责任。 有时你需要违反它。 以下是Google针对MVP的示例:

TaskDetailPresenter调用ShowEditTask ,它负责在TaskDetailFragment打开一个新的Activity

但是你也可以使用CommandPattern ,这是一种更好的方法

interface NavigationCommand {
    void navigate();
}

因此,Presenter将在需要时使用它。

正如我在对已接受答案的评论中所写的那样,我认为从视图层管理导航是明确打破关注点分离规则:视图应该只包含更新当前UI屏幕的方法。

问题源于android平台设计,因为ActivityFragment类包含在UI屏幕上操作的方法以及发送启动其他活动(如startActivity intent对象。

解决这个问题的一种简洁方法是创建一些Navigator接口,该接口将包含与导航相关的方法,make活动实现它并将其注入到演示者中。 这种方式至少可以从演示者的立场导航和UI操作中分离出来。 然而,从活动的角度来看,它可能看起来很奇怪:现在他们经常实现两个接口(Navigator和View)并将它们的引用传递给演示者2次。 如果由于这个原因您决定从视图层管理导航,那么至少将导航方法与操作UI的方法分开:永远不要在同一方法中执行导航和UI操作。

在我看来,如果你从视图层打开一个活动会更好。 我更喜欢Presenter尽可能少地了解Activity。

如果存在应该启动哪些活动的某些条件,您可以使用以下内容:

public class Presenter {

    private ViewsPresentation mViewsPresentation;

    public void someButtonClicked() {
        if (/*some condition*/) {
            mViewsPresentation.startFirstActivity();
        } else {
            mViewsPresentation.startSecondActivity();
        }
    }

    public interface ViewsPresentation {
        void startFirstActivity();
        void startSecondActivity();
    }

}

我已经做了这个解决方案(在 Kotlin 中):
我创建了一个名为 ViewNavigator 的界面

interface ViewNavigator {
    fun navigateTo(target: Class<*>)
}

然后我让视图接口实现它

interface View : ViewNavigator {
    //...
}

然后实际视图(活动)可以覆盖navigateTo函数

override fun navigateTo(target: Class<*>) {
    startActivity(Intent(this, target))
}

因此,无论何时我想导航到任何活动,我都可以简单地将其写在演示者类中。 例如:

override fun onAnimationFinished() {
    view.navigateTo(HomeActivity::class.java)
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM