AndroidX下使用Activity和Fragment的變化詳解
過(guò)去的一段時(shí)間,AndroidX 軟件包下的 Activity/Fragmet 的 API 發(fā)生了很多變化。讓我們看看它們是如何提升Android 的開(kāi)發(fā)效率以及如何適應(yīng)當(dāng)下流行的編程規(guī)則和模式。
本文中描述的所有功能現(xiàn)在都可以在穩(wěn)定的 AndroidX 軟件包中使用,它們?cè)谌ツ昃寻l(fā)布或移至穩(wěn)定版本。
在構(gòu)造器中傳入布局 ID
從 AndroidX AppCompat 1.1.0 和 Fragment 1.1.0 ( 譯者注:AppCompat 包含 Fragment,且 Fragment 包含 Activity,詳情見(jiàn)【整理】Jetpack 主要組件的依賴(lài)及傳遞關(guān)系 )開(kāi)始,您可以使用將 layoutId 作為參數(shù)的構(gòu)造函數(shù):
class MyActivity : AppCompatActivity(R.layout.my_activity) class MyFragmentActivity: FragmentActivity(R.layout.my_fragment_activity) class MyFragment : Fragment(R.layout.my_fragment)
這種方法可以減少 Activity/Fragment 中方法重寫(xiě)的數(shù)量,并使類(lèi)更具可讀性。無(wú)需在Activity 中重寫(xiě) onCreate() 即可調(diào)用 setContentView() 方法。另外,無(wú)需手動(dòng)在Fragment中重寫(xiě) onCreateView 即可手動(dòng)調(diào)用 Inflater 來(lái)擴(kuò)展視圖。
擴(kuò)展 Activity/Fragment 的靈活性
借助 AndroidX 新的 API ,可以減少在 Activity/Fragment 處理某些功能的情況。通常,您可以獲取提供某些功能的對(duì)象并向其注冊(cè)您的處理邏輯,而不是重寫(xiě) Activity / Fragment 中的方法。這樣,您現(xiàn)在可以在屏幕上組成幾個(gè)獨(dú)立的類(lèi),獲得更高的靈活性,復(fù)用代碼,并且通常在不引入自己的抽象的情況下,對(duì)代碼結(jié)構(gòu)具有更多控制。讓我們看看這在兩個(gè)示例中如何工作。
OnBackPressedDispatcher
有時(shí),您需要阻止用戶返回上一級(jí)。在這種情況下,您需要在 Activity 中重寫(xiě) onBackPressed()方法。但是,當(dāng)您使用 Fragment 時(shí),沒(méi)有直接的方法來(lái)攔截返回。在 Fragment 類(lèi)中沒(méi)有可用的onBackPressed() 方法,這是為了防止同時(shí)存在多個(gè) Fragment 時(shí)發(fā)生意外行為。
但是,從 AndroidX Activity 1.0.0 開(kāi)始,您可以使用 OnBackPressedDispatcher 在您可以訪問(wèn)該 Activity 的代碼的任何位置(例如,在 Fragment 中)注冊(cè) OnBackPressedCallback。
class MyFragment : Fragment() { override fun onAttach(context: Context) { super.onAttach(context) val callback = object : OnBackPressedCallback(true) { override fun handleOnBackPressed() { // Do something } } requireActivity().onBackPressedDispatcher.addCallback(this, callback) } }
您可能會(huì)在這里注意到另外兩個(gè)有用的功能:
OnBackPressedCallback 的構(gòu)造函數(shù)中的布爾類(lèi)型的參數(shù)有助于根據(jù)當(dāng)前狀態(tài)動(dòng)態(tài) 打開(kāi)/關(guān)閉按下的行為
addCallback() 方法的可選第一個(gè)參數(shù)是 LifecycleOwner,以確保僅在您的生命周期感知對(duì)象(例如,F(xiàn)ragment)至少處于 STARTED 狀態(tài)時(shí)才使用回調(diào)。
通過(guò)使用 OnBackPressedDispatcher ,您不僅可以獲得在 Activity 之外處理返回鍵的便捷方式。根據(jù)您的需要,您可以在任意位置定義 OnBackPressedCallback,使其可復(fù)用,或根據(jù)應(yīng)用程序的架構(gòu)進(jìn)行任何操作。您不再需要重寫(xiě)Activity 中的 onBackPressed 方法,也不必提供自己的抽象的來(lái)實(shí)現(xiàn)需求的代碼。
SavedStateRegistry
如果您希望 Activity 在終止并重啟后恢復(fù)之前的狀態(tài),則可能要使用 saved state 功能。過(guò)去,您需要在 Activity 中重寫(xiě)兩個(gè)方法:onSaveInstanceState 和onRestoreInstanceState。您還可以在 onCreate 方法中訪問(wèn)恢復(fù)的狀態(tài)。同樣,在 Fragment中,您可以使用onSaveInstanceState 方法(并且可以在 onCreate,onCreateView 和onActivityCreated方法中恢復(fù)狀態(tài))。
從 AndroidX SavedState 1.0.0(它是 AndroidX Activity 和 AndroidX Fragment 內(nèi)部的依賴(lài)。譯者注:您不需要單獨(dú)聲明它)開(kāi)始,您可以訪問(wèn) SavedStateRegistry,它使用了與前面描述的 OnBackPressedDispatcher 類(lèi)似的機(jī)制:您可以從 Activity / Fragment 中獲取SavedStateRegistry,然后 注冊(cè)您的 SavedStateProvider:
class MyActivity : AppCompatActivity() { companion object { private const val MY_SAVED_STATE_KEY = "my_saved_state" private const val SOME_VALUE_KEY = "some_value" } private lateinit var someValue: String private val savedStateProvider = SavedStateRegistry.SavedStateProvider { Bundle().apply { putString(SOME_VALUE_KEY, someValue) } } override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) savedStateRegistry .registerSavedStateProvider(MY_SAVED_STATE_KEY, savedStateProvider) } fun someMethod() { someValue = savedStateRegistry .consumeRestoredStateForKey(MY_SAVED_STATE_KEY) ?.getString(SOME_VALUE_KEY) ?: "" } }
如您所見(jiàn),SavedStateRegistry 強(qiáng)制您將密鑰用于數(shù)據(jù)。這樣可以防止您的數(shù)據(jù)被 attach 到同一個(gè) Activity/Fragment的另一個(gè) SavedStateProvider 破壞。就像在OnBackPressedDispatcher 中一樣,您可以例如將 SavedStateProvider 提取到另一個(gè)類(lèi),通過(guò)使用所需的任何邏輯使其與數(shù)據(jù)一起使用,從而在應(yīng)用程序中實(shí)現(xiàn)清晰的保存狀態(tài)行為。
此外,如果您在應(yīng)用程序中使用 ViewModel,請(qǐng)考慮使用 AndroidX ViewModel-SavedState 使你的ViewModel 可以保存其狀態(tài)。為了方便起見(jiàn),從 AndroidX Activity 1.1.0 和 AndroidXFragment 1.2.0 開(kāi)始,啟用 SavedState 的SavedStateViewModelFactory 是在獲取ViewModel 的所有方式中使用的默認(rèn)工廠:委托 ViewModelProvider 構(gòu)造函數(shù)和ViewModelProviders.of() 方法。
FragmentFactory
Fragment 最常提及的問(wèn)題之一是不能使用帶有參數(shù)的構(gòu)造函數(shù)。例如,如果您使用 Dagger2 進(jìn)行依賴(lài)項(xiàng)注入,則無(wú)法使用 Inject 注解 Fragment 構(gòu)造函數(shù)并指定參數(shù)?,F(xiàn)在,您可以通過(guò)指定FragmentFactory 類(lèi)來(lái)減少 Fragment 創(chuàng)建過(guò)程中的類(lèi)似問(wèn)題。通過(guò)在 FragmentManager 中注冊(cè)FragmentFactory,可以重寫(xiě)實(shí)例化 Fragment 的默認(rèn)方法:
class MyFragmentFactory : FragmentFactory() { override fun instantiate(classLoader: ClassLoader, className: String): Fragment { // Call loadFragmentClass() to obtain the Class object val fragmentClass = loadFragmentClass(classLoader, className) // Now you can use className/fragmentClass to determine your prefered way // of instantiating the Fragment object and just do it here. // Or just call regular FragmentFactory to instantiate the Fragment using // no arguments constructor return super.instantiate(classLoader, className) } }
如您所見(jiàn),該API非常通用,因此您可以執(zhí)行想要?jiǎng)?chuàng)建 Fragment 實(shí)例的所有操作?;氐?Dagger2示例,例如,您可以注入FragmentFactory Provider <Fragment> 并使用它來(lái)獲取 Fragment 對(duì)象。
測(cè)試 Fragment
從AndroidX Fragment 1.1.0 開(kāi)始,可以使用 Fragment 測(cè)試組件提供 FragmentScenario 類(lèi),該類(lèi)可以幫助在測(cè)試中實(shí)例化 Fragment 并進(jìn)行單獨(dú)測(cè)試:
// To launch a Fragment with a user interface: val scenario = launchFragmentInContainer<FirstFragment>() // To launch a headless Fragment: val scenario = launchFragment<FirstFragment>() // To move the fragment to specific lifecycle state: scenario.moveToState(CREATED) // Now you can e.g. perform actions using Espresso: onView(withId(R.id.refresh)).perform(click()) // To obtain a Fragment instance: scenario.onFragment { fragment ->
More Kotlin!
很高興看到 -ktx AndroidX 軟件包中提供了許多有用的 Kotlin 擴(kuò)展方法,并且定期添加了新的方法。例如,在AndroidX Fragment-KTX 1.2.0 中,使用片段化類(lèi)型的擴(kuò)展名可用于FragmentTransaction 上的 replace() 方法。將其與 commit() 擴(kuò)展方法結(jié)合使用,我們可以獲得以下代碼:
// Before supportFragmentManager .beginTransaction() .add(R.id.container, MyFragment::class.java, null) .commit() // After supportFragmentManager.commit { replace<MyFragment>(R.id.container) }
FragmentContainerView
一件小而重要的事情。如果您將 FrameLayout 用作 Fragment 的容器,則應(yīng)改用FragmentContainerView 。它修復(fù)了一些動(dòng)畫(huà) z軸索引順序問(wèn)題和窗口插入調(diào)度。從 AndroidXFragment 1.2.0 開(kāi)始可以使用 FragmentContainerView。
到此這篇關(guān)于AndroidX下使用Activity和Fragment的變化詳解的文章就介紹到這了,更多相關(guān)AndroidX使用Activity和Fragment內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Android入門(mén)教程之ListView的具體使用詳解
- Android notifyDataSetChanged() 動(dòng)態(tài)更新ListView案例詳解
- Android使用ExpandableListView實(shí)現(xiàn)三層嵌套折疊菜單
- Android入門(mén)教程之Fragment的具體使用詳解
- Android使用ViewPager快速切換Fragment時(shí)卡頓的優(yōu)化方案
- Android Studio使用ViewPager+Fragment實(shí)現(xiàn)滑動(dòng)菜單Tab效果
- Android ListView在Fragment中的使用示例詳解
相關(guān)文章
Android使用WebView實(shí)現(xiàn)全屏切換播放網(wǎng)頁(yè)視頻功能
這篇文章主要介紹了Android使用WebView實(shí)現(xiàn)全屏切換播放網(wǎng)頁(yè)視頻功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2019-07-07Android實(shí)現(xiàn)TV端大圖瀏覽效果的全過(guò)程
最近的開(kāi)發(fā)中遇到了個(gè)需求,需要在tv端加載很長(zhǎng)的圖片,發(fā)現(xiàn)網(wǎng)上沒(méi)有相關(guān)的資料,所以跟大家分享下,這篇文章主要給大家介紹了關(guān)于Android實(shí)現(xiàn)TV端大圖瀏覽效果的相關(guān)資料,需要的朋友可以參考下2023-01-01android將Bitmap對(duì)象保存到SD卡中的方法
這篇文章主要介紹了android將Bitmap對(duì)象保存到SD卡中的方法,涉及Android讀寫(xiě)SD卡數(shù)據(jù)的方法,需要的朋友可以參考下2015-04-04Android?studio實(shí)現(xiàn)單選按鈕
這篇文章主要為大家詳細(xì)介紹了Android?studio實(shí)現(xiàn)單選按鈕,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-05-05Android采用GET方法進(jìn)行網(wǎng)絡(luò)傳值
這篇文章主要為大家詳細(xì)介紹了Android采用GET方法進(jìn)行網(wǎng)絡(luò)傳值的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12移動(dòng)端開(kāi)發(fā)之Jetpack?Hilt技術(shù)實(shí)現(xiàn)解耦
Hilt的出現(xiàn)解決前兩點(diǎn)問(wèn)題,因?yàn)镠ilt是Dagger針對(duì)Android平臺(tái)的場(chǎng)景化框架,比如Dagger需要我們手動(dòng)聲明注入的地方,而Android聲明的地方不都在onCreate()嗎,所以Hilt就幫我們做了,除此之外還做了很多事情2023-02-02