- 系统签名
- 须要放到 /system/app里做为系统app
- 机顶盒开发,系统开发,车机开发,智能设备开发。
1.1. 在清单文件 AndroidManifest.xml 添加 android.uid.system 声明为系统应用。
1.2. 权限java<uses-permission android:name="android.permission.INSTALL_PACKAGES" /> <uses-permission android:name="android.permission.DELETE_PACKAGES" /> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" /> 再添加读写权限这个能够实现安装时读取路径 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />复制代码
1.3 总配置步骤图
android![]()
W~(LD8IUF59QKEAI8`V}S3H.png
- 以前公司开发时机器是api 17 ,获取了系统里pm的代码经过aidl实现加反射实现 这个方法也是比较正规的作法
- 去大家系统的源码里找到android.content.pm把里面的代码拷拷出来看图这几个就够了。
![]()
Q)Q5U%ZJU28KF)()UA@FQK1.png 代码段api
安装 public void installApp(final String path, final String packageNames){ File apkFile = new File(path); /* 当前app没法访问外置sd卡文件,exists()会为false因此直接抛异常最好 if(!apkFile.exists()){ sendBroadcastMsg(packageNames,false,"路径错误"); return; }*/ try { Class<?> clazz = Class.forName("android.os.ServiceManager"); Method method_getService = clazz.getMethod("getService", String.class); IBinder bind = (IBinder) method_getService.invoke(null, "package"); IPackageManager iPm = IPackageManager.Stub.asInterface(bind); iPm.installPackage(Uri.fromFile(apkFile), new IPackageInstallObserver.Stub(){ @Override public void packageInstalled(String packageName, int returnCode) throws RemoteException { } }, 2, apkFile.getName()); } catch (Exception e) { sendBroadcastMsg(packageNames,false,"安装异常"); e.printStackTrace(); } } 卸载 public void uninstallApp(String packageName){ try { Class<?> clazz = Class.forName("android.os.ServiceManager"); Method method_getService = clazz.getMethod("getService",String.class); IBinder bind = (IBinder) method_getService.invoke(null, "package"); IPackageManager iPm = IPackageManager.Stub.asInterface(bind); iPm.deletePackageAsUser(packageName,null,0,2); System.out.println("=================>>卸载成功"); } catch (Exception e) { e.printStackTrace(); System.out.println("=================>>卸载失败"); } }复制代码
1. rxjava版 安装若是不要rxjava能够new Thread但要记住线程里不要有刷新ui操做
private void install(final String packageName, final String filePath) {
Observable.create(new ObservableOnSubscribe<Integer>() {
@Override
public void subscribe(ObservableEmitter<Integer> e) throws Exception {
File file = new File(filePath);
if (filePath == null || filePath.length() == 0 || file == null) {
e.onNext(0);
return;
}
String[] args = { "pm", "install", "-r", filePath };
ProcessBuilder processBuilder = new ProcessBuilder(args);
Process process = null;
BufferedReader successResult = null;
BufferedReader errorResult = null;
StringBuilder successMsg = new StringBuilder();
StringBuilder errorMsg = new StringBuilder();
try {
process = processBuilder.start();
successResult = new BufferedReader(new InputStreamReader(process.getInputStream()));
errorResult = new BufferedReader(new InputStreamReader(process.getErrorStream()));
String s;
while ((s = successResult.readLine()) != null) {
successMsg.append(s);
}
while ((s = errorResult.readLine()) != null) {
errorMsg.append(s);
}
} catch (IOException e1) {
e1.printStackTrace();
} finally {
try {
if (successResult != null) {
successResult.close();
}
if (errorResult != null) {
errorResult.close();
}
} catch (IOException e1) {
e1.printStackTrace();
}
if (process != null) {
process.destroy();
}
}
if (successMsg.toString().contains("Success") || successMsg.toString().contains("success")) {
e.onNext(2);
} else {
e.onNext(1);
}
}
}).subscribeOn(Schedulers.newThread())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Observer<Integer>() {
@Override
public void onSubscribe(Disposable d) {
}
@Override
public void onNext(Integer value) {
if (value==2) {
//安装成功
} else {
//安装错误
}
}
@Override
public void onError(Throwable e) {
//安装错误
}
@Override
public void onComplete() {
}
});
}
2.卸载
private void uninstall(String packageName) {
try {
PackageManager pm = context.getPackageManager();
Method[] methods = pm!=null?pm.getClass().getDeclaredMethods():null;
Method mDel = null;
if (methods != null && methods.length>0) {
for (Method method : methods) {
if (method.getName().toString().equals("deletePackage")) {
mDel = method;
break;
}
}
}
if (mDel != null) {
mDel.setAccessible(true);
mDel.invoke(pm,packageName,null,0);
}
} catch (Exception e) {
e.printStackTrace();
}
}复制代码
- platform.pk8
- platform.x509.pem
- signapk.jar
- 下面是个人工具图
![]()
DW9_9VUX1Z$Q~@}$4ILJ)0H.png
- 方法一若是使用时若是缺乏类就把缺乏的类复制进去,若是参数不一样记得随机应变,通常原生系统是这 样,通常开发不会改动太大pm.2 . 方法二是比较好的方便快捷并且兼容性好点。