Android 靜默安裝和卸載的方法
本文介紹了Android 靜默安裝和卸載的方法,分享給大家,具體如下:
一. 條件
- 系統(tǒng)簽名
- 需要放到 /system/app里作為系統(tǒng)app
二. 適用環(huán)境
機頂盒開發(fā),系統(tǒng)開發(fā),車機開發(fā),智能設備開發(fā)。
三. 步驟
1. 在 AndroidManifest.xml 中
1.1. 在清單文件 AndroidManifest.xml 添加 android.uid.system 聲明為系統(tǒng)應用。
1.2. 權限
<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 總配置步驟圖

2. 代碼部分 (兩種方法都可以實現,現在項目選擇了第二種)
方法一
之前公司開發(fā)時機器是api 17 ,獲取了系統(tǒng)里pm的代碼通過aidl實現加反射實現 這個方法也是比較正規(guī)的做法
去你們系統(tǒng)的源碼里找到android.content.pm把里面的代碼拷拷出來看圖這幾個就夠了。

代碼段
安裝
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("=================>>卸載失敗");
}
}
方法二 (兼容到6.0,6.0以上沒測過,現在用的是這種沒一點問題兼容公司所以系統(tǒng),無需AIDL)
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();
}
}
3.記得簽名和打到system/app里去 前面需要的文件 下面是我們公司的文件名稱一般文件都是一樣的,對系統(tǒng)編譯后是可以獲取的。
- platform.pk8
- platform.x509.pem
- signapk.jar
下面是我的工具圖

4.可能出現的問題
- 方法一如果使用時如果缺少類就把缺少的類復制進去,如果參數不同記得隨機應變,一般原生系統(tǒng)是這 樣,一般開發(fā)不會改動太大pm.
- 方法二是比較好的方便快捷而且兼容性好點。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
21天學習android開發(fā)教程之SurfaceView
21天學習android開發(fā)教程之SurfaceView,SurfaceView由于可以直接從內存或者DMA等硬件接口取得圖像數據,因此是個非常重要的繪圖容器,操作相對簡單,感興趣的小伙伴們可以參考一下2016-02-02
Android?ViewModel創(chuàng)建不受橫豎屏切換影響原理詳解
這篇文章主要為大家介紹了Android?ViewModel創(chuàng)建不受橫豎屏切換影響原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-03-03
基于Google ML模型開發(fā)Android物體檢測應用
ML Kit是Google提供的機器學習SDK,包含了一系列預訓練模型,可以在Android和iOS應用中快速添加機器學習功能,本項目基于Google ML模型開發(fā)Android物體檢測應用,首先對圖像中的物體進行分類檢測,獲取分類物體的位置區(qū)域,然后結合圖像標記,逐個獲取單個物體的標簽2024-07-07
SurfaceView開發(fā)[捉小豬]手機游戲 (一)
這篇文章主要介紹了用SurfaceView開發(fā)[捉小豬]手機游戲 (一)本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-08-08
Android實現系統(tǒng)狀態(tài)欄的隱藏和顯示功能
這篇文章主要介紹了Android實現系統(tǒng)狀態(tài)欄的隱藏和顯示功能,文中還給大家?guī)硭姆N方法,大家可以根據自己需要參考下2018-07-07

