今天碰到个问题,想获取某个已安装的包的大小,没找到合适的方法。搜索了一下,发现PackageManager里面有个getPackageSizeInfo方法,可惜是hide的,而且它执行之后,会将结果回调给IPackageStatsObserver的onGetStatsCompleted方法。后来想直接计算/data/app和/system/app里面的apk大小,可是有时候会碰到权限问题,需要root才可以获取大小。 再后来,我想起系统的设置里面有一个应用程序管理,它里面列出了所有程序的占用空间大小、数据大小和缓存大小。恩,这个就是突破口。
以前写过一篇获取其他包的Context
,这个东西是真有用,这个结合反射,可以做很多神奇的事情,比如今天的这个。
上代码:
- 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");
- }
- }
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了。
1
顶
顶
0
踩
踩
评论
sizeObserverClass.getMethod("invokeGetSize", String.class,
CountDownLatch.class).invoke(soObject, packageName, new CountDownLatch(1));
因为缺少访问包的权限,所以在程序中引用“访问包”权限,即如下所示:
<uses-permission android:name="android.permission.GET_PACKAGE_SIZE"/>
这是可以程序可以正常运行,但是一直没有回调产生,也就是说下面这句一直没有调用
public void handleMessage(Message msg) {
if(msg.what == 1) {
//此处获取到PackageStats对象
ps = (PackageStats) msg.getData().getParcelable("ApplicationPackageStats");
Log.d("", ""+ps.codeSize);
}
}
经过查看代码后,通过下面的处理能得到包的大小的信息:
sizeObserverClass.getMethod("invokeGetSize", String.class, CountDownLatch.class)
.invoke(soObject, packageName, count);
Field f_mStats = sizeObserverClass.getDeclaredField("stats");
f_mStats.setAccessible(true);
ps= (PackageStats)f_mStats.get(soObject);
if(ps != null)
Log.d("packagename", "code size:" + ps.codeSize +" cacheSize:" +ps.cacheSize +" dataSize:" + ps.dataSize);
最后向chroya同学表达一下谢意