首页 > 解决方案 > 如何区分 Android 中的主要和次要配置文件?

问题描述

我正在尝试在我的辅助配置文件上禁​​用应用程序 OTA 期间的组件。我正在编写一个接收器来通过捕获“android.intent.action.MY_PACKAGE_REPLACED”来禁用该组件。在此处添加我的 AndroidManifest.xml 更改,

<receiver android:name=".receivers.OTAUpdateReceiver"android:exported="true">
   <intent-filter>
      <action android:name="android.intent.action.MY_PACKAGE_REPLACED" /> 
   </intent-filter>
</receiver>

连同添加这些权限,

<uses-permission android:name="android.permission.INTERACT_ACROSS_USERS" />
<uses-permission android:name="android.permission.INTERACT_ACROSS_USERS_FULL" />
<uses-permission android:name="android.permission.MANAGE_USERS" />
<uses-permission android:name="android.permission.CHANGE_COMPONENT_ENABLED_STATE" />

这是我的接收器代码,

private static final int PRIMARY_USER_ID = 0;
@Override
public void onReceive(Context context, Intent intent) {
    Log.i(TAG, "Entering here into ota update receiver");

    packageManager = context.getPackageManager();
    myComponent = new ComponentName(MY_PKG, MY_CLASS);

    if (mPackageManager.hasSystemFeature("com.abc.xyz") && !isPrimaryProfile()) {
        packageManager.setComponentEnabledSetting(
                myComponent,
                PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
                PackageManager.DONT_KILL_APP);
    }
}
public boolean isPrimaryProfile() {

    int userId = ActivityManagerNative.getDefault().getCurrentUser().id;
    return (userId == PRIMARY_USER_ID) ? true : false;
}

当我收到这个 MY_PACKAGE_REPLACED 意图时,我面临以下运行时异常,

E/AndroidRuntime(13492):致命异常:主 E/AndroidRuntime(13492):进程:com.abc.project,PID:13492 E/AndroidRuntime(13492):java.lang.RuntimeException:无法启动接收器 com.abc。 project.receivers.OTAUpdateReceiver: java.lang.SecurityException: PermissionDenial: getCurrentUser() from pid=13492, uid=1032112 requires android.permission.INTERACT_ACROSS_USERS E/AndroidRuntime(13492): at android.app.ActivityThread.handleReceiver(ActivityThread.java :2776) E/AndroidRuntime(13492): 在 android.app.ActivityThread.access$1700(ActivityThread.java:160) E/AndroidRuntime(13492): 在 android.app.ActivityThread$H.handleMessage(ActivityThread.java:1398) E/AndroidRuntime(13492): 在 android.os.Handler.dispatchMessage(Handler.java:102) E/AndroidRuntime(13492): 在 android.os.Looper.loop(Looper.java:135) E/AndroidRuntime(13492): 在 android.app.ActivityThread.main(ActivityThread.java:5597) E/AndroidRuntime(13492): 在 java.lang.reflect.Method.invoke(Native Method) E/AndroidRuntime(13492) ): 在 java.lang.reflect.Method.invoke(Method.java:372) E/AndroidRuntime(13492): 在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:984) E/AndroidRuntime (13492): 在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:779) E/AndroidRuntime(13492): 由: java.lang.SecurityException: Permission Denial:getCurrentUser() from pid=13492, uid=1032112 需要 android.permission.INTERACT_ACROSS_USERS E/AndroidRuntime(13492):在 android.os.Parcel.readException(Parcel.java:1566) E/AndroidRuntime(13492):在 android.os.Parcel.readException(Parcel.java :1519) E/AndroidRuntime(13492): 在 android.app。ActivityManagerProxy.getCurrentUser(ActivityManagerNative.java:4816) E/AndroidRuntime(13492): 在

尽管此权限“INTERACT_ACROSS_USERS”是在 AndroidManifest.xml 中定义的,但它没有得到反映。我错过了什么吗?或者有没有其他方法可以实现这一目标?

我的主要目标是区分主要和次要配置文件或确保用户不在主要配置文件中。

标签: androidandroid-intentandroid-permissionsandroid-runtimeandroid-user-profile

解决方案


能够在 UserManager 的getSerialNumberForUser api的帮助下区分主要和次要配置文件,

public boolean isPrimaryProfile(final Context context) {
        UserHandle userHandle = Process.myUserHandle();
        UserManager userManager = (UserManager) context.getSystemService(Context.USER_SERVICE);
        return (userManager != null) ? userManager.getSerialNumberForUser(userHandle) == PRIMARY_USER_ID : false;
}

对于主要用户/所有者,PRIMARY_USER_ID 的值为 0。


推荐阅读