adb可用于授予各种权限,但此权限很特殊,并且通过root将其用于此权限似乎无法正常运行:
pm grant packageName android.permission.PACKAGE_USAGE_STATS
操作系统说它被授予,但应用程序本身无法检测到它。
请参阅该应用的代码以查看它:
class MainActivity : AppCompatActivity() {
companion object {
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
@JvmStatic
fun hasPermission(context: Context): Boolean {
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP)
return false
val appOps = context.getSystemService(Context.APP_OPS_SERVICE) as AppOpsManager
val mode = appOps.checkOpNoThrow(AppOpsManager.OPSTR_GET_USAGE_STATS, android.os.Process.myUid(), context.getPackageName())
val granted = if (mode == AppOpsManager.MODE_DEFAULT) {
(context.checkCallingOrSelfPermission(android.Manifest.permission.PACKAGE_USAGE_STATS) == PackageManager.PERMISSION_GRANTED)
} else {
(mode == AppOpsManager.MODE_ALLOWED)
}
return granted
// val appOps = context.getSystemService(Context.APP_OPS_SERVICE) as AppOpsManager
// val granted = appOps.checkOpNoThrow(AppOpsManager.OPSTR_GET_USAGE_STATS, android.os.Process.myUid(), context.packageName) == AppOpsManager.MODE_ALLOWED
// return granted
}
}
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
button.setOnClickListener {
val granted = hasPermission(this@MainActivity)
Toast.makeText(this@MainActivity, "has permission:" + granted, Toast.LENGTH_SHORT).show()
}
button2.setOnClickListener {
startActivity(Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS))
}
// extra code for granting the app the permission using root
}
}
清单:
<manifest package="com.example.user.myapplication" xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools">
<uses-permission
android:name="android.permission.PACKAGE_USAGE_STATS" tools:ignore="ProtectedPermissions"/>
<application
android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round" android:supportsRtl="true" android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN"/>
<category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
</application>
</manifest>
布局文件
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
android:layout_height="match_parent" android:gravity="center" android:orientation="vertical"
tools:context=".MainActivity">
<TextView
android:layout_width="wrap_content" android:layout_height="wrap_content"
android:text="check the permission using below buttons before and after you grant it via root"/>
<Button
android:id="@+id/button" android:layout_width="wrap_content" android:layout_height="wrap_content"
android:text="check permission via code"/>
<Button
android:id="@+id/button2" android:layout_width="wrap_content" android:layout_height="wrap_content"
android:text="go to settings screen"/>
</LinearLayout>
为什么我通过PC使用adb,它工作正常,但如果我在应用程序本身中使用root,我得到一个奇怪的结果,操作系统说应用程序获得了许可,但应用程序无法使用发现它?
我应该怎么做才能在应用程序中使用root真正授予权限,并正确检测它?
答案 0 :(得分:1)
使用adb shell pm grant [PACKAGE_NAME] android.permission.PACKAGE_USAGE_STATS
在多个设备上为我工作。
确保将以下内容添加到AndroidManifest
:
<uses-permission android:name="android.permission.PACKAGE_USAGE_STATS"
tools:ignore="ProtectedPermissions"/>
我也可以使用root来授予权限。同样,应用必须在AndroidManifest
。
您可以使用AndroidShell以root身份运行命令。例如:
String packageName = context.getPackageName();
CommandResult result = Shell.SU.run("pm grant " + packageName + " android.permission.PACKAGE_USAGE_STATS");
正如Sagar所述,您还可以要求用户在设置应用中授予权限,这是Google建议的执行此操作的方式:
startActivity(new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS));