【Android应用开发】-(18)静默方式实现批量安装卸载应用程序_EvoShark的博客-程序员宅基地

技术标签: Android  

     前段时间做了一个批量安装卸载应用程序的小应用,由于安装卸载应用程序的部分API是隐藏的,所以必须在ubuntu下下载Android系统源码,并编译之后使用MM命令编译生成APK文件,其实也难。思路是这样的,在XX/packages/apps目录下有一个PackageInstaller的应用程序,Android机器中安装卸载都是由这个应用程序完成的。但是它没有批量安装和卸载的功能,如果要在自己的应用程序中添加批量安装和卸载的功能,其实很简单,只需要参考PakcageInstaller里面的安装卸载代码加个循环就可以了。但值得注意的是在编译的过程中必须复制PackageInstaller里面的Android.mk文件,修改文件为工程目录名。好了,废话不再多说,下面是关键代码

   1、 Android.mk文件

  1. LOCAL_PATH:= $(call my-dir)  
  2. include $(CLEAR_VARS)  
  3.   
  4. LOCAL_MODULE_TAGS := optional  
  5.   
  6. LOCAL_SRC_FILES := $(call all-subdir-java-files)  
  7.   
  8. LOCAL_PACKAGE_NAME := PackageInstaller  
  9. LOCAL_CERTIFICATE := platform  
  10.   
  11. include $(BUILD_PACKAGE)  


   2、PakcageInstaller.java文件(关键代码)

  1. package cn.ceadic.apkmgr;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileNotFoundException;  
  5. import java.io.FileOutputStream;  
  6. import java.io.IOException;  
  7.   
  8. import android.content.Context;  
  9. import android.content.Intent;  
  10. import android.content.pm.PackageInfo;  
  11. import android.content.pm.PackageManager;  
  12. import android.content.pm.PackageManager.NameNotFoundException;  
  13. import android.net.Uri;  
  14. import android.util.Log;  
  15.   
  16. import android.content.pm.IPackageInstallObserver;  
  17. import android.content.pm.IPackageDeleteObserver;  
  18. import android.os.FileUtils;  
  19.   
  20.   
  21. public class PackageInstaller {  
  22.       
  23.     private File mTmpFile;  
  24.     private final String TMP_FILE_NAME = "tmpCopy.apk";  
  25.   
  26.     private final static String TAG = "PackInstaller";  
  27.     private Context mContext;  
  28.   
  29.     public PackageInstaller(Context context) {  
  30.         mContext = context;  
  31.     }  
  32.   
  33.       
  34.     public void install(String path,String packageName){  
  35.          Intent intent = new Intent(Intent.ACTION_VIEW);  
  36.          intent.setDataAndType(Uri.fromFile(new File(path)),  
  37.          "application/vnd.android.package-archive");  
  38.          mContext.startActivity(intent);  
  39.     }  
  40.       
  41.     public void instatllBatch(String path, String packageName) {  
  42.   
  43.         Log.i(TAG, "path=" + path);  
  44.         int installFlags = 0;  
  45.         PackageManager pm = mContext.getPackageManager();  
  46.         try {  
  47.             PackageInfo pi = pm.getPackageInfo(packageName,  
  48.                     PackageManager.GET_UNINSTALLED_PACKAGES);  
  49.             if (pi != null) {  
  50.                 installFlags |= PackageManager.INSTALL_REPLACE_EXISTING;  
  51.             }  
  52.         } catch (NameNotFoundException e) {  
  53.         }  
  54.         if ((installFlags & PackageManager.INSTALL_REPLACE_EXISTING) != 0) {  
  55.             Log.w(TAG, "Replacing package:" + packageName);  
  56.         }  
  57.   
  58.         // Create temp file before invoking install api  
  59.         mTmpFile = createTempPackageFile(path);  
  60.         if (mTmpFile == null) {  
  61.             // Message msg = mHandler.obtainMessage(INSTALL_COMPLETE);  
  62.             // msg.arg1 = PackageManager.INSTALL_FAILED_INSUFFICIENT_STORAGE;  
  63.             // mHandler.sendMessage(msg);  
  64.             return;  
  65.         }  
  66.         Uri mPackageURI = Uri.parse("file://" + mTmpFile.getPath());  
  67.         String installerPackageName = mContext.getIntent().getStringExtra(  
  68.                 Intent.EXTRA_INSTALLER_PACKAGE_NAME);  
  69.   
  70.         PackageInstallObserver observer = new PackageInstallObserver();  
  71.         pm.installPackage(mPackageURI, observer, installFlags,  
  72.                 installerPackageName);  
  73.     }  
  74.   
  75.     private File createTempPackageFile(String filePath) {  
  76.         File tmpPackageFile = mContext.getFileStreamPath(TMP_FILE_NAME);  
  77.         if (tmpPackageFile == null) {  
  78.             Log.w(TAG, "Failed to create temp file");  
  79.             return null;  
  80.         }  
  81.         if (tmpPackageFile.exists()) {  
  82.             tmpPackageFile.delete();  
  83.         }  
  84.         // Open file to make it world readable  
  85.         FileOutputStream fos;  
  86.         try {  
  87.             fos = openFileOutput(TMP_FILE_NAME, MODE_WORLD_READABLE);  
  88.         } catch (FileNotFoundException e1) {  
  89.             Log.e(TAG, "Error opening file " + TMP_FILE_NAME);  
  90.             return null;  
  91.         }  
  92.         try {  
  93.             fos.close();  
  94.         } catch (IOException e) {  
  95.             Log.e(TAG, "Error opening file " + TMP_FILE_NAME);  
  96.             return null;  
  97.         }  
  98.   
  99.         File srcPackageFile = new File(filePath);  
  100.         if (!FileUtils.copyFile(srcPackageFile, tmpPackageFile)) {  
  101.             Log.w(TAG, "Failed to make copy of file: " + srcPackageFile);  
  102.             return null;  
  103.         }  
  104.         return tmpPackageFile;  
  105.     }  
  106.   
  107.     private class PackageInstallObserver extends IPackageInstallObserver.Stub {  
  108.         public void packageInstalled(String packageName, int returnCode) {  
  109.             // Message msg = mHandler.obtainMessage(INSTALL_COMPLETE);  
  110.             // msg.arg1 = returnCode;  
  111.             // mHandler.sendMessage(msg);  
  112.             Log.i(TAG, "====INSTALL_COMPLETE");  
  113.         }  
  114.     }  
  115.       
  116.     private class PackageDeleteObserver extends IPackageDeleteObserver.Stub {  
  117.         public void packageDeleted(boolean succeeded) {  
  118. //            Message msg = mHandler.obtainMessage(UNINSTALL_COMPLETE);  
  119. //            msg.arg1 = succeeded?SUCCEEDED:FAILED;  
  120. //            mHandler.sendMessage(msg);  
  121.             Log.i(TAG, "====UNINSTALL_COMPLETE");  
  122.         }  
  123.     }  
  124.       
  125.     public void uninstall(String packageName){  
  126.         Uri packageURI = Uri.parse("package:" + packageName);  
  127.         Intent uninstallIntent = new Intent(Intent.ACTION_DELETE,  
  128.         packageURI);  
  129.         mContext.startActivity(uninstallIntent);  
  130.     }  
  131.       
  132.     public void uninstallBatch(String packageName) {  
  133.         PackageDeleteObserver observer = new PackageDeleteObserver();  
  134.         mContext.getPackageManager().deletePackage(packageName, observer, 0);  
  135.           
  136.     }  
  137. }  


   3、别忘记添加权限

  1. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>  
  2.     <uses-permission android:name="android.permission.INSTALL_PACKAGES" />  
  3.     <uses-permission android:name="android.permission.DELETE_PACKAGES" />  
  4.     <uses-permission android:name="android.permission.CLEAR_APP_CACHE" />  
  5.     <uses-permission android:name="android.permission.READ_PHONE_STATE" />  
  6.     <uses-permission android:name="android.permission.CLEAR_APP_USER_DATA" />  

以上代码在Android2.1的SDK中编译通过,并正确批量安装卸载应用程序

转载自:http://blog.csdn.net/tangcheng_ok/article/details/6681453
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/EvoShark/article/details/8470488

智能推荐

如何画好架构图_爱编程_的博客-程序员宅基地

导读:技术传播的价值,不仅仅体现在通过商业化产品和开源项目来缩短我们构建应用的路径,加速业务的上线速率,也体现在优秀工程师在工作效率提升、产品性能优化和用户体验改善等经验方面的分享,以提...

js函数求斐波那契数列Fibonacci中的第n个数_每一天,每一步的博客-程序员宅基地_js斐波那契数列函数第n个数

// 求斐波那契数列Fibonacci中的第n个数是多少? 1 1 2 3 5 8 13 21...function getFib(n) { var n1 = 1; var n2 = 1; for (var i = 2; i &lt; n; i++) { // for (var i = 3; i &lt;= n; i++) { n3 = n1 + n2; n1 = n2; n2 = n3; } return n3;}var n = getFib(3);console.log(

java.lang.NullPointerException org.apache.jsp.login_jsp._jspInit(login_jsp.java:25)_iyangPengYu的博客-程序员宅基地

exceptionorg.apache.jasper.JasperException: java.lang.NullPointerException org.apache.jasper.servlet.JspServletWrapper.handleJspException(JspServletWrapper.java:538) org.apache.jasper.servlet.Jsp...

postgresql 11 最佳实践 运行参数_数据库人生的博客-程序员宅基地

os: centos 7.6db: postgresql 11最佳实践参数如下,部分参数需要根据实际情况调整shared_buffersarchive_commandcluster_name#------------------------------------------------------------------------------# CONNECTIONS AND AUTHENTICATION#-----------------------------------------

R数据分析:样本量计算的底层逻辑与实操,pwr包_公众号Codewar原创作者的博客-程序员宅基地_样本量计算 r方

样本量问题真的是好多人的老大难,是很多同学科研入门第一个拦路虎,今天给本科同学改大创标书又遇到这个问题,我想想不止是本科生对这个问题不会,很多同学从上研究生到最后脱离科研估计也没能把这个问题弄得很明白,那么希望大伙儿在看了这篇文章能够更加深入地理解样本量计算的逻辑,也能对大家的科研设计中的样本量设计部分有所启发。样本量计算的逻辑还记得我们最开始接触统计推断的时候,大家都知道一个词叫做原假设,原假设一般来讲都是“阴性的”,我们统计推断要做的事情便是推翻原假设从而得出有“统计学意义的结果”,怎么去推翻?

串口485接法图_rs485接口接线方法_weixin_39543758的博客-程序员宅基地

展开全部RS485采用差分信号负逻辑,+2V~+6V表示“0”,- 6V~- 2V表示“1”。62616964757a686964616fe4b893e5b19e31333365643661RS485有两线制和四线制两种接线,四线制是全双工通讯方式,两线制是半双工通讯方式。具体操作如下。1.RS485和RS232一样都是基于串口的通讯接口,数据收发的操作是一致的,所以使用的是同样WinCE的底层驱...

随便推点

性能测试工具大全_thomashtq的博客-程序员宅基地

Performance test tools (53 found)AllmonDescription:The main goal of the project is to create a distributed generic system collecting and storing various runtime metrics collections used

overridePendingTransition的使用_Errol_King的博客-程序员宅基地_overridepending

Android中不同Activity之间的切换是不可避免的事情,Android中提供了一个方法来解决这个问题,即overridePendingTransition(A,B)函数此方法在startActivity()或者是finish()后调用,例如startActivity(new Intent(MainActivity.this,SecondActivity.class));overrid...

Selenium 八种元素定位方法_一个程序媛~的博客-程序员宅基地

# coding = utf-8from time import sleepfrom selenium import webdriver# 驱动文件路径driverfile_path = r'E:\pycharm-workspace\0-testData\chromedriver.exe'# 启动浏览器driver = webdriver.Chrome(executable_path=driverfile_path)driver.implicitly_wait(30)# 打开百度首页d.

JavaScript中的图形可视化库_xfxf996的博客-程序员宅基地

I have a data structure that represents a directed graph, and I want to render that dynamically on

javax.xml + java.io xml和java对象互转工具类_MemoryMaple的博客-程序员宅基地

import javax.xml.bind.JAXBContext;import javax.xml.bind.JAXBException;import javax.xml.bind.Marshaller;import javax.xml.bind.Unmarshaller;import java.io.StringReader;import java.io.StringWriter;...

OpenCv cv::Mat用法_岳飞传的博客-程序员宅基地

OpenCv cv::Mat类用法1转载地址:http://www.360doc.com/content/13/0415/16/10724725_278435687.shtml1、使用准备:using  namespace  cv;2、Mat的声明Mat m=Mat(rows, cols, type);Mat m=Mat(Size(width,hei

推荐文章

热门文章

相关标签