单 Activity+ 多 Fragment 模式
自从晓得这一招之后我根本不太违心应用 activity 了,fragment 能够疾速创立和治理,能够正当设计页面跳转,设计炫酷的跳转动画,一些操作能够对立进行治理。
- 用 Fragment 代替 Activity
以前大部分时候都是将 Activity 作为页面,Fragment 作为页面中的子页面(过后称之为碎片),基本上大部分性能由 activity 实现,比方老版本的淘宝 app 就是有上百个 activity,过后卡顿的不要不要的。随着技术迭代,咱们发现 activtiy 创立、切换、销毁所耗费的性能远比 fragment 要大,fragment 现在也能代替 activity 实现大部分性能。
- 将 Activity 作为容器
我了解的 单 Activity+ 多 Fragment
模式并不是指一个 App 肯定只有一个 activity,对于一些业务相干的场景,能够整合成一个 单 Activity+ 多 Fragment
模块,将 activity 作为 fragment 的容器,让 fragment 去做 UI 绘制工作。
- 治理 Fragment 栈
咱们能够应用 navigation 治理 fragment,fragment 之间的跳转、栈治理都轻而易举,navigation 还能够设置切换动画、页面间的数据传递。
Navigation 组件
Navigation 是 Jetpack 组件之一,很早之前 iOS 就是采纳的这种跳转形式,过后就在想 Android 为啥没有,没多久 Navigation 就面世了。
Navigation 能够了解为以一个治理 fragment 的容器,在容器中各个 fragment 能够实现任意跳转,
根底应用:
- 咱们须要在布局中创立 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"/>
- 创立 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>
- 应用 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_home
、navi_collection
、navi_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