• 博客园logo
  • 会员
  • 周边
  • 新闻
  • 博问
  • 闪存
  • 众包
  • 赞助商
  • Chat2DB
    • 搜索
      所有博客
    • 搜索
      当前博客
  • 写随笔 我的博客 短消息 简洁模式
    用户头像
    我的博客 我的园子 账号设置 会员中心 简洁模式 ... 退出登录
    注册 登录
lamber
博客园    首页    新随笔       管理    订阅  订阅

Android获取应用程序的大小

                                                        Android获取应用程序的大小  

                                                                                                                                                                     2010-12-29 10:11:36

今天碰到个问题,想获取某个已安装的包的大小,没找到合适的方法。搜索了一下,发现PackageManager里面有个getPackageSizeInfo方法,可惜是hide的,而且它执行之后,会将结果回调给IPackageStatsObserver的onGetStatsCompleted方法。后来想直接计算/data/app和/system/app里面的apk大小,可是有时候会碰到权限问题,需要root才可以获取大小。        再后来,我想起系统的设置里面有一个应用程序管理,它里面列出了所有程序的占用空间大小、数据大小和缓存大小。恩,这个就是突破口。
       以前写过一篇获取其他包的Context ,这个东西是真有用,这个结合反射,可以做很多神奇的事情,比如今天的这个。

       上代码:

Java代码 
  1. package chroya.demo;  
  2.   
  3. import java.lang.reflect.Constructor;  
  4. import java.lang.reflect.Field;  
  5. import java.lang.reflect.InvocationTargetException;  
  6. import java.util.concurrent.CountDownLatch;  
  7.   
  8. import android.app.Activity;  
  9. import android.content.Context;  
  10. import android.content.pm.PackageStats;  
  11. import android.content.pm.PackageManager.NameNotFoundException;  
  12. import android.os.Bundle;  
  13. import android.os.Handler;  
  14. import android.os.Message;  
  15. import android.util.Log;  
  16.   
  17. public class Main extends Activity {  
  18.     private PackageStats ps;  
  19.       
  20.     public void getPackageStats(String packageName) {  
  21.         try {  
  22.             //获取setting包的的Context  
  23.             Context mmsCtx = createPackageContext("com.android.settings",  
  24.                     Context.CONTEXT_INCLUDE_CODE | Context.CONTEXT_IGNORE_SECURITY);  
  25.             //使用setting的classloader加载com.android.settings.ManageApplications类  
  26.             Class<?> maClass = Class.forName("com.android.settings.ManageApplications", true, mmsCtx.getClassLoader());  
  27.             //创建它的一个对象  
  28.             Object maObject = maClass.newInstance();  
  29.               
  30.             /* 
  31.              * 将私有域mPm赋值。因为mPm在SizeObserver的invokeGetSize中用到了, 
  32.              * 却因为没有执行onCreate而没有初始化,所以要在此处初始化。 
  33.              */  
  34.             Field f_mPm = maClass.getDeclaredField("mPm");  
  35.             f_mPm.setAccessible(true);              
  36.             f_mPm.set(maObject, mmsCtx.getPackageManager());  
  37.               
  38.             /* 
  39.              * 给mHandler赋值为重新定义的Handler,以便接收SizeObserver的 
  40.              * onGetStatsCompleted回调方法中dispatch的消息,从中取PackageStats对象。 
  41.              * */  
  42.             Field f_mHandler = maClass.getDeclaredField("mHandler");  
  43.             f_mHandler.setAccessible(true);  
  44.             f_mHandler.set(maObject, new Handler() {  
  45.                   public void handleMessage(Message msg) {  
  46.                       if(msg.what == 1) {  
  47.                           //此处获取到PackageStats对象  
  48.                           ps = (PackageStats) msg.getData().getParcelable("ApplicationPackageStats");                           
  49.                           Log.d("", ""+ps.codeSize);                            
  50.                       }  
  51.                   }  
  52.             });  
  53.               
  54.             //加载内部类SizeObserver  
  55.             Class<?> sizeObserverClass = Class.forName("com.android.settings.ManageApplications$SizeObserver", true, mmsCtx.getClassLoader());  
  56.             Constructor sizeObserverConstructor = sizeObserverClass.getDeclaredConstructors()[0];  
  57.             sizeObserverConstructor.setAccessible(true);  
  58.             /* 
  59.              * 创建SizeObserver对象,两个参数,第一个是外部类的对象, 
  60.              * 也就是ManageApplications对象,第二个是msgId,也就是 
  61.              * 分发消息的id,跟Handler接收的msgId一样。 
  62.              * */  
  63.             Object soObject = sizeObserverConstructor.newInstance(maObject, 1);  
  64.             //执行invokeGetSize方法  
  65.             sizeObserverClass.getMethod("invokeGetSize", String.class,  
  66.                     CountDownLatch.class).invoke(soObject, packageName, new CountDownLatch(1));           
  67.         } catch (NameNotFoundException e) {  
  68.             e.printStackTrace();  
  69.         } catch (ClassNotFoundException e) {  
  70.             e.printStackTrace();  
  71.         } catch (IllegalAccessException e) {  
  72.             e.printStackTrace();  
  73.         } catch (IllegalArgumentException e) {  
  74.             e.printStackTrace();  
  75.         } catch (SecurityException e) {  
  76.             e.printStackTrace();  
  77.         } catch (InvocationTargetException e) {  
  78.             e.printStackTrace();  
  79.         } catch (NoSuchMethodException e) {  
  80.             e.printStackTrace();  
  81.         } catch (InstantiationException e) {  
  82.             e.printStackTrace();  
  83.         } catch (NoSuchFieldException e) {  
  84.             e.printStackTrace();  
  85.         }  
  86.     }  
  87.       
  88.     @Override  
  89.     public void onCreate(Bundle savedInstanceState) {  
  90.         super.onCreate(savedInstanceState);    
  91.         getPackageStats("chroya.demo");         
  92.     }  
  93. }  

 
       注释都在代码里面了,稍微理解一下应该都能懂的。
       获取到PackageStats对象,就可以从中获取到应用程序的占用空间大小、数据大小和缓存大小。

 

      另,这毕竟只是hack code,不可能通用。这段代码的局限性是,只有1.5能用,而且如果别人把setting包去掉了,也没法使用。要写出各版本SDK通用的代码,就必须查看每个版本的setting包,看代码有何变化,然后根据上面给出的思路为每个版本写一个方法,就ok了。

 
想要获得成功,首先要自己相信自己,再者要赢得周围朋友的信任!
posted @ 2012-05-04 23:12  android5k  阅读(1964)  评论(0)    收藏  举报
刷新页面返回顶部
博客园  ©  2004-2026
浙公网安备 33010602011771号 浙ICP备2021040463号-3