Android获取应用程序大小的方法

所属分类: 软件编程 / Android 阅读数: 1395
收藏 0 赞 0 分享

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

上代码:

Java代码

复制代码 代码如下:

package chroya.demo; 

import java.lang.reflect.Constructor; 
import java.lang.reflect.Field; 
import java.lang.reflect.InvocationTargetException; 
import java.util.concurrent.CountDownLatch; 

import android.app.Activity; 
import android.content.Context; 
import android.content.pm.PackageStats; 
import android.content.pm.PackageManager.NameNotFoundException; 
import android.os.Bundle; 
import android.os.Handler; 
import android.os.Message; 
import android.util.Log; 

public class Main extends Activity { 
    private PackageStats ps; 

    public void getPackageStats(String packageName) { 
        try { 
            //获取setting包的的Context 
            Context mmsCtx = createPackageContext("com.android.settings", 
                    Context.CONTEXT_INCLUDE_CODE | Context.CONTEXT_IGNORE_SECURITY); 
            //使用setting的classloader加载com.android.settings.ManageApplications类 
            Class<?> maClass = Class.forName("com.android.settings.ManageApplications", true, mmsCtx.getClassLoader()); 
            //创建它的一个对象 
            Object maObject = maClass.newInstance(); 

            /*
             * 将私有域mPm赋值。因为mPm在SizeObserver的invokeGetSize中用到了,
             * 却因为没有执行onCreate而没有初始化,所以要在此处初始化。
             */ 
            Field f_mPm = maClass.getDeclaredField("mPm"); 
            f_mPm.setAccessible(true);             
            f_mPm.set(maObject, mmsCtx.getPackageManager()); 

            /*
             * 给mHandler赋值为重新定义的Handler,以便接收SizeObserver的
             * onGetStatsCompleted回调方法中dispatch的消息,从中取PackageStats对象。
             * */ 
            Field f_mHandler = maClass.getDeclaredField("mHandler"); 
            f_mHandler.setAccessible(true); 
            f_mHandler.set(maObject, new Handler() { 
                  public void handleMessage(Message msg) { 
                      if(msg.what == 1) { 
                          //此处获取到PackageStats对象 
                          ps = (PackageStats) msg.getData().getParcelable("ApplicationPackageStats");                          
                          Log.d("", ""+ps.codeSize);                           
                      } 
                  } 
            }); 

            //加载内部类SizeObserver 
            Class<?> sizeObserverClass = Class.forName("com.android.settings.ManageApplications$SizeObserver", true, mmsCtx.getClassLoader()); 
            Constructor sizeObserverConstructor = sizeObserverClass.getDeclaredConstructors()[0]; 
            sizeObserverConstructor.setAccessible(true); 
            /*
             * 创建SizeObserver对象,两个参数,第一个是外部类的对象,
             * 也就是ManageApplications对象,第二个是msgId,也就是
             * 分发消息的id,跟Handler接收的msgId一样。
             * */ 
            Object soObject = sizeObserverConstructor.newInstance(maObject, 1); 
            //执行invokeGetSize方法 
            sizeObserverClass.getMethod("invokeGetSize", String.class, 
                    CountDownLatch.class).invoke(soObject, packageName, new CountDownLatch(1));          
        } catch (NameNotFoundException e) { 
            e.printStackTrace(); 
        } catch (ClassNotFoundException e) { 
            e.printStackTrace(); 
        } catch (IllegalAccessException e) { 
            e.printStackTrace(); 
        } catch (IllegalArgumentException e) { 
            e.printStackTrace(); 
        } catch (SecurityException e) { 
            e.printStackTrace(); 
        } catch (InvocationTargetException e) { 
            e.printStackTrace(); 
        } catch (NoSuchMethodException e) { 
            e.printStackTrace(); 
        } catch (InstantiationException e) { 
            e.printStackTrace(); 
        } catch (NoSuchFieldException e) { 
            e.printStackTrace(); 
        } 
    } 

    @Override 
    public void onCreate(Bundle savedInstanceState) { 
        super.onCreate(savedInstanceState);   
        getPackageStats("chroya.demo");        
    } 

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

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

想要获得成功,首先要自己相信自己,再者要赢得周围朋友的信任!

更多精彩内容其他人还在看

android byte[] 和short[]转换的方法代码

这篇文章主要介绍了android byte[] 和short[]转换的方法代码,有需要的朋友可以参考一下
收藏 0 赞 0 分享

Android获取应用程序大小的方法

这篇文章主要介绍了Android获取应用程序大小的方法,有需要的朋友可以参考一下
收藏 0 赞 0 分享

Android获取其他包的Context实例代码

这篇文章主要介绍了Android获取其他包的Context实例代码,有需要的朋友可以参考一下
收藏 0 赞 0 分享

Android放大镜的实现代码

这篇文章主要介绍了Android放大镜的实现代码,有需要的朋友可以参考一下
收藏 0 赞 0 分享

Android 读取Properties配置文件的小例子

这篇文章主要介绍了Android 读取Properties配置文件的小例子,有需要的朋友可以参考一下
收藏 0 赞 0 分享

Android通讯录开发之删除功能的实现方法

这篇文章主要介绍了Android通讯录开发之删除功能的实现方法,有需要的朋友可以参考一下
收藏 0 赞 0 分享

使用ViewPager实现android软件使用向导功能实现步骤

现在的大部分android软件,都是使用说明,就是第一次使用该软件时,会出现向导,可以左右滑动,然后就进入应用的主界面了,下面我们就实现这个功能
收藏 0 赞 0 分享

android在异步任务中关闭Cursor的代码方法

android在异步任务中如何关闭Cursor?在我们开发应用的时候,很多时候会遇到这种问题,下面我们就看看代码如何实现
收藏 0 赞 0 分享

Android自定义桌面功能代码实现

android自定义桌面其实很简单,看一个例子就明白了
收藏 0 赞 0 分享

android将图片转换存到数据库再从数据库读取转换成图片实现代码

有时候我们想把图片存入到数据库中,尽管这不是一种明智的选择,但有时候还是不得以会用到,下面说说将图片转换成byte[]数组存入到数据库中去,并从数据库中取出来转换成图像显示出来
收藏 0 赞 0 分享
查看更多