• 转载 Android获取应用程序的大小 (使用反射)


           今天碰到个问题,想获取某个已安装的包的大小,没找到合适的方法。搜索了一下,发现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. }  
    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
    分享到:
    评论
    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  
    相当不错,赞一个!
  • 相关阅读:
    写在vue总结之前(一)
    前端应该掌握的web基础和网络知识
    sass之为什么要使用预处理器
    ThinkPHP简单的验证码实现
    ajax接收php返回得到一堆html代码
    Bootstrap 4,“未捕获错误:Bootstrap工具提示需要Tether(http://github.hubspot.com/tether/)”
    百度AI开放平台- API实战调用
    最短路径算法—Dijkstra(迪杰斯特拉)算法分析与实现(C/C++)
    C#避免踩坑之如何添加paint事件
    php插入mysql中文数据出现乱码
  • 原文地址:https://www.cnblogs.com/shallwe99/p/2468994.html
Copyright © 2020-2023  润新知