`
chroya
  • 浏览: 662248 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

Android获取应用程序的大小

阅读更多

       今天碰到个问题,想获取某个已安装的包的大小,没找到合适的方法。搜索了一下,发现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");       
    }
}

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

 

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

2
0
分享到:
评论
2 楼 sudongfeng 2010-12-02  
在android2.1下面,运行到下面一句会产生异常:
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同学表达一下谢意
1 楼 freespace521 2010-10-29  
相当不错,赞一个!

相关推荐

    android获取应用程序大小,缓存数据信息

    1. **获取应用程序大小**: - **APK大小**:可以通过读取AndroidManifest.xml文件中的`&lt;manifest package="..."&gt;`标签来获取包名,然后结合系统的`PackageInfo`类获取到安装的APK的大小。使用`PackageManager`的`...

    Android获取应用程序的包大小\缓存大小\数据大小

    在Android开发中,有时我们需要获取应用程序的包大小、缓存大小以及数据大小,这有助于我们进行性能优化、存储管理或者用户信息展示。本教程将详细讲解如何利用反射和AIDL技术来实现这一功能。 首先,我们需要理解...

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

    综上所述,Android获取应用程序大小的方法涉及了反射、隐藏API的使用、异步回调处理和权限管理等多个技术点。在实际开发中,应谨慎考虑安全性和兼容性,尽量避免直接操作系统组件,而是优先选择官方提供的API。

    Android获取应用程序大小和缓存的实例代码

    以上就是Android获取应用程序大小和缓存的实例代码的核心部分。通过这样的实现,开发者可以轻松地列出设备上所有应用的大小和缓存信息,从而帮助用户管理和清理存储空间。这个功能在系统优化工具或者第三方应用管理...

    Android 获取应用程序的大小,数据大小,缓存大小

    在Android开发中,有时我们需要获取应用程序的大小,包括它的安装包大小、数据大小以及缓存大小,以便于分析应用的存储占用情况或者优化资源管理。本文将深入探讨如何实现这一功能,并结合源码分析和使用工具的方法...

    Android开发,获取当前手机安装的所有应用、根据包名获取应用信息、获取当前应用程序的包名、获取程序 图标等关于安装在手机上的

    Android开发,获取当前手机安装的所有应用、根据包名获取应用信息、获取当前应用程序的包名、获取程序 图标、获取程序的版本号、获取程序的名字、获取程序的权限、获取程序的签名、获取当前包名的 SHA1、将获取到得...

    Android中获取应用程序(包)的大小-----PackageManager的使用(二)

    本文将详细讲解如何通过Android的`PackageManager`服务来获取应用程序的大小,以及背后的实现原理。 首先,我们需要理解`PackageManager`在Android中的角色。`PackageManager`是Android系统的一个核心组件,它负责...

    获取应用程序大小

    在Android开发中,获取应用程序的大小是一个常见的需求,这有助于我们了解应用的资源占用情况,优化性能,或者在用户界面中提供相关信息。本项目提供的源码将帮助开发者实现这一功能。下面,我们将深入探讨如何在...

    android获取应用程序信息小Demo

    在Android开发中,获取应用程序的信息是一项常见的任务,这有助于开发者监控应用的状态,或者为用户提供关于应用的详细信息。本教程将通过一个小型的Demo来演示如何获取应用的名称、版本、缓存大小以及应用程序的...

    Android中获取应用程序(包)的信息 PackageManager的使用

    本文将详细讲解如何利用`PackageManager`来获取Android应用程序(包)的相关信息。 首先,让我们理解`PackageManager`的基本概念。它是Android系统服务的一部分,允许应用程序访问和操作其他已安装的应用程序的元...

    Android中获取应用程序(包)的信息-PackageManager的使用方法

    ### Android中获取应用程序(包)的信息—PackageManager的使用方法 #### 概述 在Android开发过程中,有时我们需要获取系统中应用程序的信息,例如包名(`packagename`)、标签(`label`)、图标(`icon`)以及占用...

    Android获取当前应用分配的最大内存和目前使用内存的方法

    Android 原生系统一般默认16M,但是国内手机一般都是特殊定制的,都有修改系统的内存大小,所有有时候,要查看具体应用系统分配的内存大小,还是需要实际去测试的, 测试方法如下: 方式一: ActivityManager ...

    android获取手机内存大小

    在Android平台上,获取手机内存大小是一项常见的需求,无论是为了优化应用程序性能、监控设备资源使用情况,还是为了提供用户关于设备状态的信息。以下是一份详细的指南,涵盖了如何在Android中获取内存信息。 首先...

    Android如何获得网络资源的大小

    本文将详细介绍如何在Android应用程序中获取网络资源的实际大小。 #### 一、使用`HttpURLConnection`获取资源大小 在Android中,最常用的方法之一是通过`HttpURLConnection`来获取远程资源的信息。下面是一个具体...

    Android如何动态调整应用字体大小详解

    Android 应用程序中,字体大小的调整是一个非常重要的设置,特别是在面向中老年客户的项目中,自带的字体无法满足客户需求的情况下。通常情况下, Android 应用字体大小默认随系统设置的字体大小而变化,但有时我们...

    Android 获取应用缓存大小与清除缓存的方法

    在Android开发中,管理和清理应用缓存是保持应用程序性能和用户设备存储空间的重要步骤。`FileCacheUtils` 类提供了一套方法来实现这一目标。以下是对这个类中各个方法的详细解释: 1. **cleanInternalCache...

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

    在Android平台上,获取应用程序的大小通常涉及到访问系统级的API,因为这些信息是私有的,普通应用无法直接访问。为了实现这一目标,我们需要利用AIDL(Android Interface Definition Language)和Java的反射机制。...

    安卓SD卡文件管理更新信息相关-android获取应用程序缓存并计算大小清除缓存.zip

    这篇内容主要探讨了如何在Android平台上获取应用的缓存文件、计算它们的大小以及如何清除这些缓存。首先,我们来看看“JavaApk源码说明.txt”,这个文件可能包含了关于源码的详细解释和实现步骤。 在Android中,每...

Global site tag (gtag.js) - Google Analytics