乐趣区

关于android:写一个MVVM快速开发框架谈一谈单Activity多Fragment模式

单 Activity+ 多 Fragment 模式

自从晓得这一招之后我根本不太违心应用 activity 了,fragment 能够疾速创立和治理,能够正当设计页面跳转,设计炫酷的跳转动画,一些操作能够对立进行治理。

  1. 用 Fragment 代替 Activity

以前大部分时候都是将 Activity 作为页面,Fragment 作为页面中的子页面(过后称之为碎片),基本上大部分性能由 activity 实现,比方老版本的淘宝 app 就是有上百个 activity,过后卡顿的不要不要的。随着技术迭代,咱们发现 activtiy 创立、切换、销毁所耗费的性能远比 fragment 要大,fragment 现在也能代替 activity 实现大部分性能。

  1. 将 Activity 作为容器

我了解的 单 Activity+ 多 Fragment模式并不是指一个 App 肯定只有一个 activity,对于一些业务相干的场景,能够整合成一个 单 Activity+ 多 Fragment模块,将 activity 作为 fragment 的容器,让 fragment 去做 UI 绘制工作。

  1. 治理 Fragment 栈

咱们能够应用 navigation 治理 fragment,fragment 之间的跳转、栈治理都轻而易举,navigation 还能够设置切换动画、页面间的数据传递。

Navigation 组件

Navigation 是 Jetpack 组件之一,很早之前 iOS 就是采纳的这种跳转形式,过后就在想 Android 为啥没有,没多久 Navigation 就面世了。

Navigation 能够了解为以一个治理 fragment 的容器,在容器中各个 fragment 能够实现任意跳转,

根底应用:

  1. 咱们须要在布局中创立 Fragment 容器:
    <androidx.fragment.app.FragmentContainerView
        android:id="@+id/main_fragment_container"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:name="androidx.navigation.fragment.NavHostFragment"
        app:defaultNavHost="true"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:navGraph="@navigation/nav_main"/>
  1. 创立 navigation.xml 文件
<?xml version="1.0" encoding="utf-8"?>
<navigation xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/navigation_main"
    app:startDestination="@+id/mainFragment">

    <fragment
        android:id="@+id/mainFragment"
        android:name="com.example.mvvm_develop.MainFragment"
        android:label="MainFragment" />


</navigation>
  1. 应用 NavController
val navController = (childFragmentManager.findFragmentById(R.id.module_fragment_container) as NavHostFragment).navController
// 跳转
navController.navigate(R.id.mainFragment)

一些具体参数和用法:

navGraph

这个值指向 xml 文件,在 xml 文件中咱们能够定义 fragment,跳转行为,目的地等。

创立、新增 Fragment:

创立跳转行为:

动画、目的地、返回栈配置:

NavController

字面意思就是导航控制器,NavController 能够管制跳转、返回、动画、监听等操作。咱们能够应用它进行灵便的跳转,Google 还出了一些 Navigation Demo 演示如何配合 Toolbar 和底部导航栏进行应用。

对于具体的用法这里不解说了,很多文章都有,也能够参考官网。

Navigation 存在的问题:

重走生命周期

Navigation 目前有个问题:Fragment 回退重走生命周期,这个问题可能是 Google 想让 Fragment 和 activity 领有同样的工作模式,单重走生命周期真的很烦,咱们能够自定义 NavHostFragment 去修复这个问题,具体参考我的项目代码

批改之后应用如下:

    android:name="androidx.navigation.fragment.NavHostFragment"
    批改为咱们自定义的 NavHostFragment:android:name="com.example.baselibrary.navigation.NavHostFragment"
    
    <fragment
        android:id="@+id/navigation_main"
        android:name="com.example.baselibrary.navigation.NavHostFragment"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:defaultNavHost="true"
        app:navGraph="@navigation/nav_main"/>

组件化中应用 Navigation

咱们通常应用底部导航栏将 app 划分出不同的性能,这些都是独自的 module, 然而在 navigation 中怎么进行 module 间的跳转呢?

比方:

其布局文件就是一个FragmentContainerView+BottomNavigationView,切换上面按钮的时候须要切换到不同的 moduel 页面。首先咱们将不同的 moduel 视为一个“单 activity+ 多 fragment”的模块,或者也能够省略 activity。

形式一:
google 的 demo 中是在 MainActivity 中创立一个 main\_navGraph, 其中蕴含了不同子 moduel 的 navGraph , 如下:

<?xml version="1.0" encoding="utf-8"?>
<navigation xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/nav_module"
    app:startDestination="@+id/navi_home">

    <include app:graph="@navigation/navi_home"/>
    <include app:graph="@navigation/navi_collection"/>
    <include app:graph="@navigation/navi_center"/>

</navigation>

navi_homenavi_collectionnavi_center是子 moduel 中的 navGraph 文件,这种做法要求其指定startDestination,而且只能跳转到startDestination

配合 BottomNavigationView 应用:

val navController = (childFragmentManager.findFragmentById(R.id.module_fragment_container) as NavHostFragment).navController
setupWithNavController(binding.bottomNav,navController)

这样子的确能够实现 moduel 间的切换,然而我发现这种办法每次切换 naviagtion 都会从新初始化,导致性能耗费很大。

兴许是我应用姿态不对?

形式二:
因为 app module 自身是须要依赖各个子 moduel 的,咱们能够在 navGraph 间接应用子 moduel 中的 Fragment,主页面只需退出各个 module 的主 Fragment 就行了:

<?xml version="1.0" encoding="utf-8"?>
<navigation xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/nav_module"
    app:startDestination="@+id/navi_home">
    <fragment
        android:id="@+id/navi_home"
        android:name="com.xlu.module_tab1.HomeFragment"
        android:label="HomeFragment" />
    <fragment
        android:id="@+id/navi_collection"
        android:name="com.xlu.module_collection.FragmentCollection"
        android:label="CenterFragment" />
    <fragment
        android:id="@+id/navi_center"
        android:name="com.xlu.module_center.CenterFragment"
        android:label="FragmentCollection" />

</navigation>

在底部状态切换的时候间接切换 Fragment 就行了:

        val navController = (childFragmentManager.findFragmentById(R.id.module_fragment_container) as NavHostFragment).navController

        binding.bottomNav.setOnItemSelectedListener(object :NavigationBarView.OnItemSelectedListener{override fun onNavigationItemSelected(item: MenuItem): Boolean {navController.navigate(item.itemId)
                return true
            }
        })

偷懒的话能够将 BottomNavigationView 应用的 menu 中的 id 与 navGraph 中设置成一样的啊哈哈哈哈


基本上 ARouter+Navigation 能够满足大部分的页面跳转需要,但还是有一些难点,就是不同 moduel 之间的 navGraph 怎么互相管制,临时没有想到太好的解决办法(能够通过之前提到的想外提供接口服务实现),毕竟 Navigation 就没打算为组件化筹备。

相干视频:
价值 100w+Android 我的项目实战大全:MVVM 详解
Android(安卓) 开发零根底从入门到精通:MVVM 实战
原文链接:https://juejin.cn/post/6997422487654891533

退出移动版