优化android14低内存设备连接蓝牙键盘/鼠标后点击Disconnect断开蓝牙连接,页面卡顿(将1180ms优化到629ms)

目录

1.背景

2.分析

3.解决方案


1.背景

在ST进行测试的过程中,发现在连接蓝牙键盘和鼠标并在设备详情界面断开连接的时候页面有卡顿现象,这里不方便展示,就是在蓝牙的设备详情界面,然后点击断开会有卡顿现象

如下是详细的描述:

在 Android 系统的蓝牙设备详情页(Settings App)中,当我们点击"断开连接(Disconnect)"按钮来断开一个**蓝牙鼠标**时,出现了严重的视觉卡顿:

  1. 手指按下按钮,Material Design 的灰色水波纹(Ripple)刚荡漾出来一半,整个画面瞬间**冻结**。

  2. 画面卡死约 1 秒钟。

  3. 1 秒后,画面恢复,按钮文字突变为"Connect(连接)"。

本文就介绍如何优化这个问题,将蓝牙断开的流程从1180ms优化到629ms,并且解决水波纹卡住问题

2.分析

其实发现在内存低的设备上整体都是比较卡卡的状态,所以有一些稍微更加卡顿的界面就会凸显出来,其实代码都是google的android 14源码,主要就是设备的区别,然后我们来分析下这种问题怎么处理

首先查看日志

日志耗时分析:

开始断开:04-09 20:14:21.610

更新已断开状态:04-09 20:14:21.976

界面刷新:04-09 20:14:22.790

上述进行总结:

点击断开到上层收到状态已断开耗时:366ms

点击断开到界面显示已断开耗时:1180ms

然后我们查询日志中的异常点,如下是在一次断开的流程中相关的异常日志

可以看出来在断开蓝牙的过程中居然界面重新启动了,这肯定是会耗时很多的,然后我们在看看有没有其他异常信息

可以看出来在测试的过程中甚至有掉帧现象,所以肯定是界面显示卡顿的,主线程有时卡顿1秒钟,这肯定肉眼看起来很卡顿

上述我们得到两个信息,

1是鼠标和键盘在断开蓝牙过程中重新启动了Activity

2是由于设备配置太低,主线程已经处理不过来突如其来的大量刷新任务,其实在加日志查看的过程中由于Settings/SystemUI设计的臃肿性导致断开蓝牙后有15个组件一起刷新,所以导致主线程根本处理不过来

3.解决方案

首要解决的就是在断开蓝牙过程中Activity重启现象,在AndroidManifest中对应的Activity添加如下标识符,让其在蓝牙断开连接过程中不会重新启动Activity

android:configChanges="orientation|screenSize|keyboard|keyboardHidden|navigation"

这样我们的耗时过长问题基本解决了,但是发现还是会卡顿,水波纹显示不出来,其实从日志中这时我们从日志中查看onCreate的打印没有了,但是还是会打印出来

Skipped 48 frames! The application may be doing too much work on its main thread.

其实上述虽然我们让其不重新启动了,但是臃肿(15个组件一起更新)的问题还是存在的,所以此时我们需要排优先级,先把点击断开过程的水波纹显示出来,然后开始进行断开流程

这里我们用view.postDelay方式即可

此时优化的差不多了,然后我想进一步优化,就把一些可以放在子线程的方法给放到子线程中了,如下是最终的patch

复制代码
diff --git a/qssi14/frameworks/base/packages/SettingsLib/src/com/android/settingslib/bluetooth/LocalBluetoothProfileManager.java b/qssi14/frameworks/base/packages/SettingsLib/src/com/android/settingslib/bluetooth/LocalBluetoothProfileManager.java
index 803ad1c..848c989 100644
--- a/qssi14/frameworks/base/packages/SettingsLib/src/com/android/settingslib/bluetooth/LocalBluetoothProfileManager.java
+++ b/qssi14/frameworks/base/packages/SettingsLib/src/com/android/settingslib/bluetooth/LocalBluetoothProfileManager.java
@@ -50,6 +50,7 @@
 
 import com.android.internal.util.ArrayUtils;
 import com.android.internal.util.CollectionUtils;
+import com.android.settingslib.utils.ThreadUtils;
 
 import java.lang.reflect.Constructor;
 import java.lang.reflect.InvocationTargetException;
@@ -399,17 +400,19 @@
         }
 
         public void onReceive(Context context, Intent intent, BluetoothDevice device) {
-            if (device == null) {
-                Log.w(TAG, "StateChangedHandler receives state-change for invalid device");
-                return;
-            }
+            ThreadUtils.postOnBackgroundThread(() -> {
+                if (device == null) {
+                    Log.w(TAG, "StateChangedHandler receives state-change for invalid device");
+                    return;
+                }
 
-            CachedBluetoothDevice cachedDevice = mDeviceManager.findDevice(device);
-            if (cachedDevice == null) {
-                Log.w(TAG, "StateChangedHandler found new device: " + device);
-                cachedDevice = mDeviceManager.addDevice(device);
-            }
-            onReceiveInternal(intent, cachedDevice);
+                CachedBluetoothDevice cachedDevice = mDeviceManager.findDevice(device);
+                if (cachedDevice == null) {
+                    Log.w(TAG, "StateChangedHandler found new device: " + device);
+                    cachedDevice = mDeviceManager.addDevice(device);
+                }
+                onReceiveInternal(intent, cachedDevice);
+            });
         }
 
         protected void onReceiveInternal(Intent intent, CachedBluetoothDevice cachedDevice) {
diff --git a/qssi14/packages/apps/Settings/AndroidManifest.xml b/qssi14/packages/apps/Settings/AndroidManifest.xml
index bc116f1..2492c81 100644
--- a/qssi14/packages/apps/Settings/AndroidManifest.xml
+++ b/qssi14/packages/apps/Settings/AndroidManifest.xml
@@ -247,7 +247,8 @@
                   android:exported="false"
                   android:theme="@style/Theme.SubSettings"
                   android:knownActivityEmbeddingCerts="@array/config_known_host_certs"
-                  android:taskAffinity="com.android.settings.root" />
+                  android:taskAffinity="com.android.settings.root"
+                  android:configChanges="orientation|screenSize|keyboard|keyboardHidden|navigation" />
 
         <activity android:name=".Settings$CreateShortcutActivity"
                   android:exported="true"
@@ -4533,6 +4534,7 @@
         <activity android:name="Settings$BluetoothDeviceDetailActivity"
                   android:label="@string/device_details_title"
                   android:exported="true"
+                  android:configChanges="orientation|screenSize|keyboard|keyboardHidden|navigation"
                   android:permission="android.permission.BLUETOOTH_CONNECT">
             <intent-filter android:priority="1">
                 <action android:name="com.android.settings.BLUETOOTH_DEVICE_DETAIL_SETTINGS" />
diff --git a/qssi14/packages/apps/Settings/src/com/android/settings/bluetooth/BluetoothDetailsButtonsController.java b/qssi14/packages/apps/Settings/src/com/android/settings/bluetooth/BluetoothDetailsButtonsController.java
index 2ade4ec..c48c8ee 100644
--- a/qssi14/packages/apps/Settings/src/com/android/settings/bluetooth/BluetoothDetailsButtonsController.java
+++ b/qssi14/packages/apps/Settings/src/com/android/settings/bluetooth/BluetoothDetailsButtonsController.java
@@ -25,6 +25,7 @@
 import com.android.settings.R;
 import com.android.settingslib.bluetooth.CachedBluetoothDevice;
 import com.android.settingslib.core.lifecycle.Lifecycle;
+import com.android.settingslib.utils.ThreadUtils;
 import com.android.settingslib.widget.ActionButtonsPreference;
 
 /**
@@ -72,9 +73,14 @@
                         .setButton2Text(R.string.bluetooth_device_context_disconnect)
                         .setButton2Icon(R.drawable.ic_settings_close)
                         .setButton2OnClickListener(view -> {
-                            mMetricsFeatureProvider.action(mContext,
-                                    SettingsEnums.ACTION_SETTINGS_BLUETOOTH_DISCONNECT);
-                            mCachedDevice.disconnect();
+                            view.postDelayed(() -> {
+                                view.jumpDrawablesToCurrentState();
+                                mActionButtons.setButton2Enabled(false);
+                                ThreadUtils.postOnBackgroundThread(() -> {
+                                    mMetricsFeatureProvider.action(mContext, SettingsEnums.ACTION_SETTINGS_BLUETOOTH_DISCONNECT);
+                                    mCachedDevice.disconnect();
+                                });
+                            }, 150);
                         });
                 mConnectButtonInitialized = true;
             }
相关推荐
智塑未来15 小时前
2026办公编程显示器优选,明基4款RD按工位挑选
计算机外设
andy_haiying19 小时前
Akko 空影 TAN8 复合碳纤维电竞鼠标:碳为刃,影为锋
计算机外设
leo__5201 天前
C# 虚拟键盘(软键盘)实现
单片机·c#·计算机外设
猫头虎2 天前
猫头虎AI分享|樱桃键盘Ctrl键失效解决方案:FN+PAUSE 长按10秒恢复出厂设置保姆级教程
网络·网络协议·tcp/ip·计算机外设·键盘·机械键盘·ctrl
橡木树的叶子2 天前
STM32单片机+RGB三色灯模块+七色灯颜色数据发送到串口调试助手+源代码(使用cubuemx+vscode实现操作)
stm32·单片机·计算机外设
开开心心_Every3 天前
解决打印机共享难题的实用工具
linux·b树·安全·游戏·随机森林·pdf·计算机外设
道法自然,人法天3 天前
屏幕卡死无法点击?只用键盘重启电脑
计算机外设·电脑
崇山峻岭之间3 天前
单片机USB 鼠标键盘实验
单片机·嵌入式硬件·计算机外设
0X784 天前
金乌 Codex 桌宠 v0.1:一只会盯着鼠标看的小太阳神鸟
计算机外设
开开心心就好4 天前
无弹窗不更新的PC本地播放工具
运维·科技·macos·docker·计算机外设·ocr·powerpoint