Android普通应用升级为系统应用并获取系统权限的操作

有时候使用某些api需要使用系统权限,如调用PackageInstaller的相关接口,需要android.permission.INSTALL_PACKAGES权限,该权限系统只会授权给系统应用,此时可以考虑将我们的应用升级为系统应用,升级为系统应用有两种方法:

1、将apk放到/system/app目录下,重启手机即可,此方法比较粗暴,而且需要修改/system目录的读写权限,因此需要root,而且随着Android系统版本对权限管理越来越严,root和修改读写权限更繁琐和复杂

2、添加sharedUserId="android.uid.system"配置,同时使用系统签名对apk进行重签名,安装后即有系统权限,此方法需要获取到系统签名,但通常厂商开发的rom都不会公开签名,因此如果针对某些自己集成系统的需求,可以使用此方案。

下面针对第2种方案进行讲解。

1、首先需要在menifest文件中添加sharedUserId="android.uid.system"配置,如下

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="com.google.example"
      android:sharedUserId="android.uid.system">

2、添加需要的使用的系统权限,如

<uses-permission android:name="android.permission.INSTALL_PACKAGES" />

3、使用系统签名对apk进行签名,首先需要打未签名的包,Android Studio环境下点击右侧Gradle按钮,弹出目录,打开Tasks->build->assemble...,其中assemble开头的几项即是打包任务,双击相应项即可获取相应的包

接着要获取系统签名,找到系统签名文件“platform.pk8”和“platform.x509.pem”,在系统源码路径下

签名文件路径:android/build/target/product/security/

准备好签名工具:“signapk.jar”

位置:android/prebuilts/sdk/tools/lib

将未签名的apk、签名文件和签名工具放在同一路径下,开启终端,cd到该目录,运行如下命令即可得到带系统签名的apk

java -jar signapk.jar platform.x509.pem platform.pk8 Demo.apk signedDemo.apk

以上的方法相对还是比较繁琐,而且不便于调试,可以将系统签名打入keystore,使用该keystore即可在Android Studio环境下生成带系统签名的apk

1、首先需要生成keystore文件,生成过程在此省略,可自行百度

2、下载keytool-importkeypair,https://github.com/getfatday/keytool-importkeypair/

3、该操作需要linux系统,并安装jdk,win10环境下可以开启“基于Lunix的Windows子系统”,不需要去折腾安装虚拟机或者系统。当如,如果兄dei你已经有Linux系统或者是mac,那么恭喜你,省了很多需要折腾的步骤

4、将keystore、platform.pk8、platform.x509.pem、keytool-importkeypair放在同一目录下,开启终端,cd到此目录,运行如下命令即可将系统签名打入keystore

/keytool-importkeypair -k ./demo.keystore -p password -pk8 platform.pk8 -cert platform.x509.pem -alias demoAlias

解释:-P后接keystore密码,-alias后接key alias

5、运行结束即会在当前目录下生成带有系统签名的keystore,使用此keystore安装Android Studio打包流程打包即可得到带系统权限的apk

补充知识:Android 系统级应用守护进程

我就废话不多说了,还是直接看代码吧!

--- a/frameworks/base/packages/SystemUI/src/com/android/systemui/SystemUIApplication.java
+++ b/frameworks/base/packages/SystemUI/src/com/android/systemui/SystemUIApplication.java
@@ -16,24 +16,51 @@
package com.android.systemui;
+import android.app.ActivityManager;
+import android.app.ActivityManager.RunningAppProcessInfo;
import android.app.Application;
import android.app.Instrumentation;
import android.content.BroadcastReceiver;
import android.content.Context;
+import android.content.ComponentName;
import android.content.Intent;
import android.content.IntentFilter;
import android.content.res.Configuration;
+import android.database.ContentObserver;
+import android.net.Uri;
+import android.os.Build;
+import android.os.Handler;
import android.os.Process;
import android.os.PowerManager;
import android.os.SystemClock;
import android.os.SystemProperties;
import android.os.UserHandle;
import android.util.Log;
-
+import java.util.List;
+import java.util.Calendar;
import com.android.systemui.stackdivider.Divider;
import java.util.HashMap;
import java.util.Map;
+import java.util.Objects;
+import java.io.BufferedReader;
+import java.io.File;
+import java.io.FileReader;
+import java.io.FileWriter;
+import java.io.IOException;
+import java.text.SimpleDateFormat;
+import java.util.Date;
+import java.util.Locale;
+
+import java.io.FileInputStream;
+import java.io.FileNotFoundException;
+import java.io.FileOutputStream;
+import java.io.InputStream;
+import java.io.PrintWriter;
+import java.math.BigInteger;
+import java.security.MessageDigest;
+import java.security.NoSuchAlgorithmException;
+import java.util.Random;
/**
 * Application class for SystemUI.
@@ -42,6 +69,8 @@ public class SystemUIApplication extends Application {
   private static final String TAG = "SystemUIService";
   private static final boolean DEBUG = false;
+
+    private Context mContext;
   /**
   * The classes of the stuff to start.
@@ -86,6 +115,8 @@ public class SystemUIApplication extends Application {
     // application theme in the manifest does only work for activities. Keep this in sync with
     // the theme set there.
     setTheme(R.style.systemui_theme);
+
+        mContext = getApplicationContext();
     SystemUIFactory.createFromConfig(this);
@@ -143,6 +174,7 @@ public class SystemUIApplication extends Application {
    private static final String SETTING_SYSTEM = "tchip.provider.setting.system";
    private static final String SETTING_SECURE = "tchip.provider.setting.secure";
    private static final String SETTING_GLOBAL = "tchip.provider.setting.global";
+    private static final String BROADCAST_SCREENSHOT = "rk.android.screenshot.action";
    private String valueContent = "0";
    private MainReceiver mainReceiver;

@@ -150,6 +182,10 @@ public class SystemUIApplication extends Application {
        android.util.Log.i("AZ", log);
    }

+    private void LDLog(String log) {
+        android.util.Log.i("LD", log);
+    }
+
    public class MainReceiver extends BroadcastReceiver {
        @Override
@@ -225,6 +261,18 @@ public class SystemUIApplication extends Application {
                intentDown.putExtra("android.intent.extra.KEY_CONFIRM", false);
                intentDown.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                context.startActivity(intentDown);
+            } else if (BROADCAST_SCREENSHOT.equals(action)) {
+                Intent intentScreenShot = new Intent("android.intent.action.SCREENSHOT");
+                intentScreenShot.putExtra("path", intentScreenShot.getExtras().getString("path"));
+                intentScreenShot.putExtra("name", intentScreenShot.getExtras().getString("name"));
+                context.sendBroadcast(intentScreenShot);
+            } else if (BROADCAST_TIMETICK.equals(action)) {
+                //checkLaidianAppAlive();
+                try{
+                    mHandler.removeCallbacks(laidianAppR);
+                }catch(Exception e){
+                }
+                mHandler.postDelayed(laidianAppR, 1000);
            }
        }
    }
@@ -326,4 +374,82 @@ public class SystemUIApplication extends Application {
   public SystemUI[] getServices() {
     return mServices;
   }
+
+
+    Handler mHandler = new Handler();
+    private int laidianAppCheckTime = 3000;
+  Runnable laidianAppR = new Runnable(){
+
+        @Override
+        public void run() {
+           checkAppAlive(pkgName", "clsName");
+            mHandler.postDelayed(laidianAppR, laidianAppCheckTime);
+        }
+
+  };
+
+    private void checkAppAlive(String pkgName, String clsName){
+        long startTime = System.currentTimeMillis();
+        if(!isAppAlive(pkgName)){
+            LDLog("checkAppAlive.start to start app... ");
+            try{
+                ComponentName componentName = new ComponentName(pkgName, clsName);
+        Intent intent = new Intent();
+        intent.setComponent(componentName);
+        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
+        mContext.startActivity(intent);
+            }catch(Exception e){
+
+            }
+        }else{
+            LDLog("checkAppAlive.app is running... ");
+        }
+        LDLog("checkAppAlive.used time : " + (System.currentTimeMillis() - startTime));
+    }
+
+    public boolean isAppAlive(String packageName){
+        //boolean isAlive = false;
+        //String command = "ps | grep " + packageName/* + " | busybox awk '{print $2}'"*/;
+        //String[] result = ExecCmd.getPackagePID(command);
+        //for(String str :result){
+      // if(!"-1".equals(str)){
+      //     isAlive = true;
+      // }
+        //}
+        //return isAlive;
+
+    // 获取正在运行的进程
+    /*ActivityManager activityManager = (ActivityManager) mContext.getSystemService(Context.ACTIVITY_SERVICE);
+    List<RunningAppProcessInfo> appProcesses = activityManager.getRunningAppProcesses();
+
+    for (RunningAppProcessInfo appProcess : appProcesses) {
+      String[] pkgList = appProcess.pkgList;
+      for (String pckName : pkgList) {
+        if(packageName.equals(pckName)){
+            return true;
+        }
+
+      }
+    }*/
+        ActivityManager am = (ActivityManager) mContext.getSystemService(Context.ACTIVITY_SERVICE);
+        List<ActivityManager.RecentTaskInfo> recentTasks = am.getRecentTasks(30, ActivityManager.RECENT_IGNORE_UNAVAILABLE
+            | ActivityManager.RECENT_INCLUDE_PROFILES);
+
+        for (ActivityManager.RecentTaskInfo recentInfo : recentTasks){
+      //if (recentInfo.origActivity != null) {
+                String pckName = recentInfo.baseIntent.getComponent().getPackageName();
+                LDLog("isAppAlive.packageName=" + pckName);
+                if(packageName.equals(pckName)){
+                    return true;
+                }
+            //}
+        }
+    return false;
+    }
+
}

以上这篇Android普通应用升级为系统应用并获取系统权限的操作就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • Android 实现永久性开启adb 的root权限

    adb 的root 权限是在system/core/adb/adb.c 中控制.主要根据ro.secure 以及 ro.debuggable 等system property 来控制. 默认即档ro.secure 为0 时,即开启root 权限,为1时再根据ro.debuggable 等选项来确认是否可以用开启root 权限.为此如果要永久性开启adb 的root 权限,有两种修改的方式: 1. 修改system property ro.secure, 让ro.secure=0. 2. 修改ad

  • Android中不支持动态申请权限的原因

    作为Android开发者,为程序增加权限是在正常不过的事情了,做法必然是在mainifest中,写入类似这样<uses-permission android:name="android.permission.INTERNET" />的信息. 以静态申请的形式来完成. 于是这里我想抛出一个问题,Android平台支持动态申请权限么. 相信很多人回答都是不支持,当然这个答案是对的,但是为什么不支持呢,知其然更要知其所以然.了解其原因还是相当有必要的. 原因列举 Android没

  • Android获取超级管理员权限的实现

    1.定义特殊的广播接收者,系统超级管理员的广播接收者 public class MyDeviceAdminReceiver extends DeviceAdminReceiver{ @Override public void onReceive(Context context,Intent intent){ //TODO } } 2.在AndroidManifest.xml文件中,注册超级管理员的广播接收者 <receiver android:name="com.example.recei

  • Android普通应用升级为系统应用并获取系统权限的操作

    有时候使用某些api需要使用系统权限,如调用PackageInstaller的相关接口,需要android.permission.INSTALL_PACKAGES权限,该权限系统只会授权给系统应用,此时可以考虑将我们的应用升级为系统应用,升级为系统应用有两种方法: 1.将apk放到/system/app目录下,重启手机即可,此方法比较粗暴,而且需要修改/system目录的读写权限,因此需要root,而且随着Android系统版本对权限管理越来越严,root和修改读写权限更繁琐和复杂 2.添加sh

  • Android获取系统储存以及内存信息的方法(一)

    众所周知,Android的手机上的信息大部分都是可以通过代码获取的,比如说爱奇艺的离线储存功能上有一个最大储存大小/剩余储存大小的功能. 获取SD卡上的储存信息: /** * 获得SD卡总大小 * * @return */ private String getSDTotalSize() { File path = Environment.getExternalStorageDirectory(); StatFs stat = new StatFs(path.getPath()); long bl

  • Android编程实现获取系统内存、CPU使用率及状态栏高度的方法示例

    本文实例讲述了Android编程实现获取系统内存.CPU使用率及状态栏高度的方法.分享给大家供大家参考,具体如下: DeviceInfoManage类用于获取系统的内存,CPU的信息,以及状态栏的高度 import java.io.BufferedReader; import java.io.FileInputStream; import java.io.FileReader; import java.io.IOException; import java.io.InputStreamReade

  • 解析Android获取系统cpu信息,内存,版本,电量等信息的方法详解

    Android获取系统cpu信息,内存,版本,电量等信息 1.CPU频率,CPU信息:/proc/cpuinfo和/proc/stat 通过读取文件/proc/cpuinfo系统CPU的类型等多种信息.读取/proc/stat 所有CPU活动的信息来计算CPU使用率 下面我们就来讲讲如何通过代码来获取CPU频率: 复制代码 代码如下: package com.orange.cpu; import java.io.BufferedReader;import java.io.FileNotFound

  • Android获取系统时间以及网络时间

    项目开发中,很多时候会用到android的时间,罗列一下获取的时间的方式,和大家共同学习进步 一.获取系统时间  1.通过Calendar类来获取系统当前的时间 Calendar calendar = Calendar.getInstance(); long unixTime = calendar.getTimeInMillis();//这是时间戳 Logger.i(TAG,"calendar--->>>"+"当前时间为:" + calendar.

  • Android系统中使用shareuserid获取系统权限的教程

    Android会为每个apk进程分配一个单独的空间(比如只能访问/data/data/自己包名下面的文件),一般情况下apk之间是禁止相互访问数据的.通过Shared User id,拥有同一个User id的多个APK可以配置成运行在同一个进程中.所以默认就是可以互相访问任意数据. 也可以配置成运行成不同的进程, 同时可以访问其他APK的数据目录下的数据库和文件.就像访问本程序的数据一样(使用IPC机制,不同进程之间,比如AIDL). 一.使用同一个shareuserid,多个apk运行到同一

  • Android 用Time和Calendar获取系统当前时间源码分享(年月日时分秒周几)

    概述 用Time和Calendar获取系统当前时间(年月日时分秒周几) 效果图 源码: import android.app.Activity; import android.os.Bundle; import android.text.format.Time; import android.view.View; import android.widget.RelativeLayout; import android.widget.TextView; import java.util.Calen

  • Android编程获取系统隐藏服务实现锁屏的方法

    本文实例讲述了Android编程获取系统隐藏服务实现锁屏的方法.分享给大家供大家参考,具体如下: 实现原理:当按锁屏键时,会发出一个广播,当界面接收到一个广播就可以实现锁频.我们可以调用IDevicePolicyManager服务中的lockNow方法来发送一个广播实现锁屏. IDevicePolicyManager是被系统隐藏掉的,需要通过反射还获取此服务. 步骤: 1.创建MyAdmin的广播接收者继承DeviceAdminReceiver 2.通过反射 ,获取IDevicePolicyMa

  • Android 调用系统相机拍摄获取照片的两种方法实现实例

    Android 调用系统相机拍摄获取照片的两种方法实现实例 在我们Android开发中经常需要做这个一个功能,调用系统相机拍照,然后获取拍摄的照片.下面是我总结的两种方法获取拍摄之后的照片,一种是通过Bundle来获取压缩过的照片,一种是通过SD卡获取的原图. 下面是演示代码: 布局文件: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http:

  • Android获取系统时间的多种方法

    Android中获取系统时间有多种方法,可分为Java中Calendar类获取,java.util.date类实现,还有android中Time实现. 现总结如下: 方法一: void getTime1(){ long time=System.currentTimeMillis();//long now = android.os.SystemClock.uptimeMillis(); SimpleDateFormat format=new SimpleDateFormat("yyyy-MM-dd

随机推荐