Android LayoutInflater.inflate耗时解决方法

本文探讨了ANR在LayoutInflater.inflate调用中的常见原因,通过深入分析View层次并提出布局优化策略,如简化层级、异步LayoutInflater和监控控件加载耗时。重点介绍了如何使用AsyncLayoutInflater和设置Factory2来避免ANR问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近项目中收到公司APM反馈的一个ANR问题,具体堆栈指向LayoutInflater.inflate。

这句原生调用怎么会发生ANR呢?

于是使用Android Studio的 Layout Inspector工具查看了下View层次,发现View层次过多,并且有复杂的自定义View,这个应该就是耗时的主因。

针对复杂的布局,我们通常的布局优化手段:简化布局的层级;include和merge结合使用;ViewStub 按需加载;减少使用wrap_content,最好使用固定宽高或者match_parent,减少计算;ConstraintLayout使用等。

另外还有其他的优化手段:如AsyncLayoutInflater异步加载布局。笔者使用这种方式解决了LayoutInflater.inflate的ANR问题。此外我们还可以在super.onCreate之前使用LayoutInflaterCompat.setFactory2来监控控件加载耗时。

异步加载布局使用如下:inflate 成功后 再调用setContentView和UI控件初始化。

 new AsyncLayoutInflater(MainActivity.this).inflate(R.layout.activity_main, null, new AsyncLayoutInflater.OnInflateFinishedListener() {
            @Override
            public void onInflateFinished(@NonNull View view, int resid, @Nullable ViewGroup parent) {
                setContentView(view);
                initUI();
            }
        });

 监控控件加载耗时:

        LayoutInflaterCompat.setFactory2(getLayoutInflater(), new LayoutInflater.Factory2() {
            @Nullable
            @Override
            public View onCreateView(@Nullable View parent, @NonNull String name, @NonNull Context context, @NonNull AttributeSet attrs) {
                long startTime = System.currentTimeMillis();
                View view = getDelegate().createView(parent, name, context, attrs);
                long cost = System.currentTimeMillis() - startTime;
                Log.d("LayoutInflateTest", "加载控件:" + name + "耗时:" + cost);
                return view;
            }

            @Nullable
            @Override
            public View onCreateView(@NonNull String name, @NonNull Context context, @NonNull AttributeSet attrs) {
                return null;
            }
        });

 

FATAL EXCEPTION: main Process: com.tplink.vigi, PID: 3108 java.lang.RuntimeException: Unable to start activity ComponentInfo{com.tplink.vigi/com.tplink.ipc.ui.deviceSetting.recordPlan.SettingRecordPlanSelectActivity}: android.view.InflateException: Binary XML file line #52 in com.tplink.vigi:layout/activity_setting_record_plan_select: Binary XML file line #52 in com.tplink.vigi:layout/activity_setting_record_plan_select: Error inflating class com.tplink.ipc.common.commonView.TPSecurTwoLineItemView at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3809) at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3971) at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:103) at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135) at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95) at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2373) at android.os.Handler.dispatchMessage(Handler.java:106) at android.os.Looper.loopOnce(Looper.java:233) at android.os.Looper.loop(Looper.java:344) at android.app.ActivityThread.main(ActivityThread.java:8200) at java.lang.reflect.Method.invoke(Native Method) at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:589) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1071) Caused by: android.view.InflateException: Binary XML file line #52 in com.tplink.vigi:layout/activity_setting_record_plan_select: Binary XML file line #52 in com.tplink.vigi:layout/activity_setting_record_plan_select: Error inflating class com.tplink.ipc.common.commonView.TPSecurTwoLineItemView Caused by: android.view.InflateException: Binary XML file line #52 in com.tplink.vigi:layout/activity_setting_record_plan_select: Error inflating class com.tplink.ipc.common.commonView.TPSecurTwoLineItemView Caused by: java.lang.reflect.InvocationTargetException at java.lang.reflect.Constructor.newInstance0(Native Method) at java.lang.reflect.Constructor.newInstance(Constructor.java:343) at android.view.LayoutInflater.createView(LayoutInflater.java:858) at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:1014) at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:965) at android.view.LayoutInflater.rInflate(LayoutInflater.java:1144) at android.view.LayoutInflater.rInflateChildren(LayoutInflater.java:1105) at android.view.LayoutInflater.rInflate(LayoutInflater.java:1147) at android.view.LayoutInflater.rInflateChildren(LayoutInflater.java:1105) at android.view.LayoutInflater.inflate(LayoutInflater.java:686) at android.view.LayoutInflater.inflate(LayoutInflater.java:538) at android.view.LayoutInflater.inflate(LayoutInflater.java:485) at androidx.appcompat.app.AppCompatDelegateImpl.setContentView(AppCompatDelegateImpl.java:775) at androidx.appcompat.app.AppCompatActivity.setContentView(AppCompatActivity.java:197) at androidx.databinding.DataBindingUtil.setContentView(DataBindingUtil.java:303) at androidx.databinding.DataBindingUtil.setContentView(DataBindingUtil.java:284) at com.tplink.ipc.ui.deviceSetting.recordPlan.SettingRecordPlanSelectActivity.initView(SettingRecordPlanSelectActivity.kt:134) at com.tplink.ipc.ui.deviceSetting.recordPlan.SettingRecordPlanSelectActivity.onCreate(SettingRecordPlanSelectActivity.kt:89) at android.app.Activity.performCreate(Activity.java:8130) at android.app.Activity.performCreate(Activity.java:8110) at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1343) at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3778) at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3971) at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:103) at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135) 2025-07-31 13:50:40.650 3108-3108 AndroidRuntime com.tplink.vigi E at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95) at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2373) at android.os.Handler.dispatchMessage(Handler.java:106) at android.os.Looper.loopOnce(Looper.java:233) at android.os.Looper.loop(Looper.java:344) at android.app.ActivityThread.main(ActivityThread.java:8200) at java.lang.reflect.Method.invoke(Native Method) at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:589) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1071) Caused by: java.lang.NullPointerException: Attempt to invoke interface method 'java.lang.Object kotlin.Lazy.getValue()' on a null object reference at com.tplink.design.list.TPTwoLineItemView.getItemViewHelper(TPListItemView.kt:618) at com.tplink.design.list.TPListItemView.setOnClickListener(TPListItemView.kt:376) at android.view.View.<init>(View.java:5840) at android.view.ViewGroup.<init>(ViewGroup.java:707) at androidx.constraintlayout.widget.ConstraintLayout.<init>(ConstraintLayout.java:598) at com.tplink.design.list.TPListItemView.<init>(TPListItemView.kt:54) at com.tplink.design.list.TPTwoLineItemView.<init>(TPListItemView.kt:614) at com.tplink.design.list.TPTwoLineItemView.<init>(TPListItemView.kt:609) at com.tplink.ipc.common.commonView.TPSecurTwoLineItemView.<init>(TPSecurTwoLineItemView.kt:13) ... 34 more
最新发布
08-01
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值