前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Android获取应用程序的大小 博客分类: Android AndroidOSF#Security

Android获取应用程序的大小 博客分类: Android AndroidOSF#Security

作者头像
chroya
发布2018-10-31 17:37:18
8540
发布2018-10-31 17:37:18
举报
文章被收录于专栏:封碎封碎

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

       上代码:

代码语言:javascript
复制
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了。

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2010-10-22 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档