savedInstanceState和 fragment.setRetainInstance以及 viewmodel的区别

本文对比分析了BundlesavedInstanceState、fragment.setRetainInstance及viewmodel在屏幕旋转时保存数据的原理与区别,深入探讨了它们在不同场景下的适用性和局限性。

以默认activity的配置 在屏幕旋转的时候,一般activty都会被重建,以这个情况为例子来说明 Bundle savedInstanceState 和 fragment.setRetainInstance 以及 viewmodel的区别

0. 转载请注明原文出处

作者github :github.com/zjw-swun 欢迎相互关注

1. 为什么要把这3个放在一块说

Bundle savedInstanceState 和 fragment.setRetainInstance 以及 viewmodel(viewModel = ViewModelProviders.of(this).get(UserProfileViewModel.class);用法而不是自己new的那种),都具备一种功能,就是当 屏幕旋转的时候( 以默认activity的配置前提),都能保存一些要被销毁掉的activity中的一些数据(如editext文本,以及recyclerView的滑动位置等),那么这3个有什么区别吗,会不会因为我们不知道原理而踩坑,下文给出答案。

2. 结论

  1. Bundle savedInstanceState 中的数据是由系统进程进行存储的,它能存储的数据容量大小有限(例如intent中如果传输Bundle内容过大会出现异常),但是比如自己app因为手机内存不足而杀掉进程的话,可以能够利用该方式进行数据还原 2.fragment.setRetainInstance 以及viewmodel(viewModel = ViewModelProviders.of(this).get(UserProfileViewModel.class);用法而不是自己new的那种)的原理是一样的,都是利用,Activity类的NonConfigurationInstances类在app进程中进行保存的,它能存储数据容量比Bundler savedInstanceState 方式要大,但是比如自己app因为手机内存不足而杀掉进程的话,则不能用该方式进行数据还原

3. 部分源码分析

以下以activity类中NonConfigurationInstances类如何在屏幕旋转的时候( 以默认activity的配置前提)如何存储并恢复NonConfigurationInstances对象为例,剖析原理 以下截图以及截图中的代码api版本为28,运行环境官方api 28的模拟器上 测试代码就不贴了很简单, 先贴一下debug断点列表,有兴趣可以试试,至于如何debug app以及debug系统进程,不知道的朋友可以看一下我另一篇文章https://www.imooc.com/article/21992 废话不多说 放图

然后看重建activity时候的断点(截图中截的是被杀死的activity 走到ondestroy的时候)

这里说明一下上一个activty先走ondrstroy 然后才重建新的activity这里和activity跳转做一个区分,然后从上图中可以发现被杀死的activity 走到ondestroy的时候中的 r和重建时候传入的参数 r(ActivityThread类中的ActivityClientRecord类型)是同一个对象,看看一下这个ActivityClientRecord类的代码

static final class ActivityClientRecord {
       //...不重要参数
        Activity.NonConfigurationInstances lastNonConfigurationInstances;
        //...不重要参数;
        }
复制代码

这就是核心的原理代码了,那么为什么说是存储在app进程中呢,根据断点列表你会发现涉及到一个关键类叫做 ActivityRelaunchItem这个类

这个类的 mActivityClientRecord就是app进程存储的那个 r了,这个 r里面包含了 lastNonConfigurationInstances,当activty切换的时候,系统进程通过binder机制通知app进程的client对象也就是activityThread间接调用 preExecute方法,进行保存,然后当重建activity的时候再系统进程通过binder机制通知app进程的client对象也就是activityThread(由activityThread.h发送 H.RELAUNCH_ACTIVITY消息通过handler机制)间接调用 execute方法 下图是activity重建时候调用 execute函数调用栈截图

重建activty后进行attach把上个被杀死的activity存下来的 rlastNonConfigurationInstances再设置给新activity

中间省略的步骤虽然多,但是根据断点列表走下来其实很清楚的。

授人以鱼不如授人以渔

对于如何调试app进程和系统进程,www.imooc.com/article/219… 一文中有具体操作,但是涉及到binder类是如何进行transact发消息给别的进程以及如何execTransact处理别的进程消息 的c++层的原理并没有给出解答,这里推荐一篇博客 blog.youkuaiyun.com/innost/arti… 讲解的还算不错。

2025-07-31 18:31:004,tiny-log,[1753957864341], [main]app crash: java.lang.RuntimeException: Unable to resume activity {com.mercusys.halo/com.tplink.mercusys.component.dashboard.view.MainActivity}: java.lang.IllegalArgumentException: View=com.android.internal.policy.DecorView{fc3d391 V.ED..... R.....ID 0,0-1024,651}[SystemTimeMerActivity] not attached to window manager at android.app.ActivityThread.performResumeActivity(ActivityThread.java:5467) at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:5500) at android.app.servertransaction.ResumeActivityItem.execute(ResumeActivityItem.java:73) at android.app.servertransaction.ActivityTransactionItem.execute(ActivityTransactionItem.java:63) at android.app.servertransaction.TransactionExecutor.executeLifecycleItem(TransactionExecutor.java:169) at android.app.servertransaction.TransactionExecutor.executeTransactionItems(TransactionExecutor.java:101) at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:80) at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2773) at android.os.Handler.dispatchMessage(Handler.java:109) at android.os.Looper.loopOnce(Looper.java:232) at android.os.Looper.loop(Looper.java:317) at android.app.ActivityThread.main(ActivityThread.java:8934) at java.lang.reflect.Method.invoke(Native Method) at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:591) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:911) Caused by: java.lang.IllegalArgumentException: View=com.android.internal.policy.DecorView{fc3d391 V.ED..... R.....ID 0,0-1024,651}[SystemTimeMerActivity] not attached to window manager at android.view.WindowManagerGlobal.findViewLocked(WindowManagerGlobal.java:587) at android.view.WindowManagerGlobal.removeView(WindowManagerGlobal.java:494) at android.view.WindowManagerImpl.removeViewImmediate(WindowManagerImpl.java:216) at android.app.Dialog.dismissDialog(Dialog.java:404) at android.app.Dialog.dismiss(Dialog.java:386) at androidx.appcompat.app.AppCompatDialog.dismiss(AppCompatDialog.java:1) at com.tplink.mercusys.component.core.lifecycle.r0.onUnregister(MeshNetworkConnectionObserver.java:21) at com.tplink.mercusys.base.lifecycle.ApplicationStateReceiverDelegate.registerMeshConnectObserver(ApplicationStateReceiverDelegate.java:15) at com.tplink.mercusys.base.lifecycle.ApplicationStateReceiverDelegate.registerMeshNetworkObserver(ApplicationStateReceiverDelegate.java:1) at com.tplink.mercusys.component.dashboard.viewmodel.BaseDashboardViewModel.x6(BaseDashboardViewModel.java:15) at com.tplink.mercusys.component.dashboard.view.NetworkTabFragment.Bg(NetworkTabFragment.java:89) at com.tplink.mercusys.component.dashboard.view.NetworkTabFragment.Ve(NetworkTabFragment.java:54) at com.tplink.apps.architecture.BaseMvvmFragment.onViewCreated(BaseMvvmFragment.kt:45) at androidx.fragment.app.Fragment.performViewCreated(Fragment.java:5) at androidx.fragment.app.FragmentStateManager.createView(FragmentStateManager.java:272) at androidx.fragment.app.FragmentStateManager.moveToExpectedState(FragmentStateManager.java:125) at androidx.fragment.app.FragmentManager.executeOpsTogether(FragmentManager.java:231) at androidx.fragment.app.FragmentManager.removeRedundantOperationsAndExecute(FragmentManager.java:82) at androidx.fragment.app.FragmentManager.execPendingActions(FragmentManager.java:22) at androidx.fragment.app.FragmentManager.dispatchStateChange(FragmentManager.java:39) at androidx.fragment.app.FragmentManager.dispatchViewCreated(FragmentManager.java:2) at androidx.fragment.app.Fragment.performViewCreated(Fragment.java:10) at androidx.fragment.app.FragmentStateManager.createView(FragmentStateManager.java:272) at androidx.fragment.app.FragmentStateManager.moveToExpectedState(FragmentStateManager.java:125) at androidx.fragment.app.FragmentManager.executeOpsTogether(FragmentManager.java:231) at androidx.fragment.app.FragmentManager.removeRedundantOperationsAndExecute(FragmentManager.java:82) at androidx.fragment.app.FragmentManager.execPendingActions(FragmentManager.java:22) at androidx.fragment.app.FragmentManager.dispatchStateChange(FragmentManager.java:39) at androidx.fragment.app.FragmentManager.dispatchResume(FragmentManager.java:12) at androidx.fragment.app.FragmentController.dispatchResume(FragmentController.java:5) at androidx.fragment.app.FragmentActivity.onResumeFragments(FragmentActivity.java:10) at androidx.fragment.app.FragmentActivity.onPostResume(FragmentActivity.java:4) at androidx.appcompat.app.AppCompatActivity.onPostResume(AppCompatActivity.java:1) at android.app.Activity.performResume(Activity.java:9266) at android.app.ActivityThread.performResumeActivity(ActivityThread.java:5457) ... 14 more 崩溃分析
最新发布
08-01
Process: com.chery.auto.media, PID: 8750 java.lang.IllegalStateException: Expected the adapter to be 'fresh' while restoring state. at androidx.viewpager2.adapter.FragmentStateAdapter.restoreState(FragmentStateAdapter.java:536) at androidx.viewpager2.widget.ViewPager2.restorePendingState(ViewPager2.java:350) at androidx.viewpager2.widget.ViewPager2.dispatchRestoreInstanceState(ViewPager2.java:375) at android.view.ViewGroup.dispatchRestoreInstanceState(ViewGroup.java:4013) at android.view.View.restoreHierarchyState(View.java:20995) at androidx.fragment.app.Fragment.restoreViewState(Fragment.java:639) at androidx.fragment.app.Fragment.restoreViewState(Fragment.java:3010) at androidx.fragment.app.Fragment.performActivityCreated(Fragment.java:3001) at androidx.fragment.app.FragmentStateManager.activityCreated(FragmentStateManager.java:580) at androidx.fragment.app.FragmentStateManager.moveToExpectedState(FragmentStateManager.java:285) at androidx.fragment.app.FragmentManager.executeOpsTogether(FragmentManager.java:2180) at androidx.fragment.app.FragmentManager.removeRedundantOperationsAndExecute(FragmentManager.java:2106) at androidx.fragment.app.FragmentManager.execPendingActions(FragmentManager.java:2002) at androidx.fragment.app.FragmentManager$5.run(FragmentManager.java:524) at android.os.Handler.handleCallback(Handler.java:938) at android.os.Handler.dispatchMessage(Handler.java:99) at android.os.Looper.loopOnce(Looper.java:201) at android.os.Looper.loop(Looper.java:288) at android.app.ActivityThread.main(ActivityThread.java:7843) at java.lang.reflect.Method.invoke(Native Method) at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:548) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1119)
07-04
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值