单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