`

Android获取其他包的Context实例,然后调用它的方法,反射!!!

 
阅读更多

Android中有Context的概念,想必大家都知道。Context可以做很多事情,打开activity、发送广播、打开本包下文件夹和数据库、获取classLoader、获取资源等等。如果我们得到了一个包的Context对象,那我们基本上可以做这个包自己能做的大部分事情。
那我们能得到吗?很高兴的告诉你,能!
Context有个createPackageContext方法,可以创建另外一个包的上下文,这个实例不同于它本身的Context实例,但是功能是一样的。

这个方法有两个参数:
1。packageName 包名,要得到Context的包名
2。flags 标志位,有CONTEXT_INCLUDE_CODE和CONTEXT_IGNORE_SECURITY两个选项。CONTEXT_INCLUDE_CODE的意思是包括代码,也就是说可以执行这个包里面的代码。CONTEXT_IGNORE_SECURITY的意思是忽略安全警告,如果不加这个标志的话,有些功能是用不了的,会出现安全警告。

下面给个小例子,执行另外一个包里面的某个类的方法。
另外一个包的包名是chroya.demo,类名Main,方法名print,代码如下:

Java代码 收藏代码
  1. package chroya.demo;
  2. import android.app.Activity;
  3. import android.os.Bundle;
  4. import android.util.Log;
  5. class Main extends Activity {
  6. @Override
  7. public void onCreate(Bundle savedInstanceState) {
  8. super.onCreate(savedInstanceState);
  9. }
  10. public void print(String msg) {
  11. Log.d("Main""msg:"+ msg);
  12. }
  13. }



本包的调用Main的print方法的代码块如下:

Java代码 收藏代码
  1. Context c = createPackageContext("chroya.demo", Context.CONTEXT_INCLUDE_CODE | Context.CONTEXT_IGNORE_SECURITY);
  2. //载入这个类
  3. Class clazz = c.getClassLoader().loadClass("chroya.demo.Main");
  4. //新建一个实例
  5. Object owner = clazz.newInstance();
  6. //获取print方法,传入参数并执行
  7. Object obj = clazz.getMethod("print", String.class).invoke(owner, "Hello");


ok,这样,我们就调用了chroya.demo包的Main类的print方法,执行结果,打印出了Hello。

怎么样,这只是一个调用其他包的代码的例子,我们获取到Context,还可以做很多事情,当然,题目所说的坏事,还是不要做为好。

 

 

Android获取应用程序的大小

2010-12-29 10:11:36| 分类: Android |字号 订阅

 

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

上代码:

Java代码
  1. package chroya.demo;
  2. import java.lang.reflect.Constructor;
  3. import java.lang.reflect.Field;
  4. import java.lang.reflect.InvocationTargetException;
  5. import java.util.concurrent.CountDownLatch;
  6. import android.app.Activity;
  7. import android.content.Context;
  8. import android.content.pm.PackageStats;
  9. import android.content.pm.PackageManager.NameNotFoundException;
  10. import android.os.Bundle;
  11. import android.os.Handler;
  12. import android.os.Message;
  13. import android.util.Log;
  14. public class Main extends Activity {
  15. private PackageStats ps;
  16. public void getPackageStats(String packageName) {
  17. try {
  18. //获取setting包的的Context
  19. Context mmsCtx = createPackageContext("com.android.settings",
  20. Context.CONTEXT_INCLUDE_CODE | Context.CONTEXT_IGNORE_SECURITY);
  21. //使用setting的classloader加载com.android.settings.ManageApplications类
  22. Class<?> maClass = Class.forName("com.android.settings.ManageApplications"true, mmsCtx.getClassLoader());
  23. //创建它的一个对象
  24. Object maObject = maClass.newInstance();
  25. /*
  26. * 将私有域mPm赋值。因为mPm在SizeObserver的invokeGetSize中用到了,
  27. * 却因为没有执行onCreate而没有初始化,所以要在此处初始化。
  28. */
  29. Field f_mPm = maClass.getDeclaredField("mPm");
  30. f_mPm.setAccessible(true);
  31. f_mPm.set(maObject, mmsCtx.getPackageManager());
  32. /*
  33. * 给mHandler赋值为重新定义的Handler,以便接收SizeObserver的
  34. * onGetStatsCompleted回调方法中dispatch的消息,从中取PackageStats对象。
  35. * */
  36. Field f_mHandler = maClass.getDeclaredField("mHandler");
  37. f_mHandler.setAccessible(true);
  38. f_mHandler.set(maObject, new Handler() {
  39. public void handleMessage(Message msg) {
  40. if(msg.what == 1) {
  41. //此处获取到PackageStats对象
  42. ps = (PackageStats) msg.getData().getParcelable("ApplicationPackageStats");
  43. Log.d(""""+ps.codeSize);
  44. }
  45. }
  46. });
  47. //加载内部类SizeObserver
  48. Class<?> sizeObserverClass = Class.forName("com.android.settings.ManageApplications$SizeObserver"true, mmsCtx.getClassLoader());
  49. Constructor sizeObserverConstructor = sizeObserverClass.getDeclaredConstructors()[0];
  50. sizeObserverConstructor.setAccessible(true);
  51. /*
  52. * 创建SizeObserver对象,两个参数,第一个是外部类的对象,
  53. * 也就是ManageApplications对象,第二个是msgId,也就是
  54. * 分发消息的id,跟Handler接收的msgId一样。
  55. * */
  56. Object soObject = sizeObserverConstructor.newInstance(maObject, 1);
  57. //执行invokeGetSize方法
  58. sizeObserverClass.getMethod("invokeGetSize", String.class,
  59. CountDownLatch.class).invoke(soObject, packageName, new CountDownLatch(1));
  60. catch (NameNotFoundException e) {
  61. e.printStackTrace();
  62. catch (ClassNotFoundException e) {
  63. e.printStackTrace();
  64. catch (IllegalAccessException e) {
  65. e.printStackTrace();
  66. catch (IllegalArgumentException e) {
  67. e.printStackTrace();
  68. catch (SecurityException e) {
  69. e.printStackTrace();
  70. catch (InvocationTargetException e) {
  71. e.printStackTrace();
  72. catch (NoSuchMethodException e) {
  73. e.printStackTrace();
  74. catch (InstantiationException e) {
  75. e.printStackTrace();
  76. catch (NoSuchFieldException e) {
  77. e.printStackTrace();
  78. }
  79. }
  80. @Override
  81. public void onCreate(Bundle savedInstanceState) {
  82. super.onCreate(savedInstanceState);
  83. getPackageStats("chroya.demo");
  84. }
  85. }


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

 

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

在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);

分享到:
评论

相关推荐

    反射获取Context的实例

    在Android开发中,反射是一种强大的编程技术,它允许我们在运行时检查和操作类、接口、字段和方法。本文将深入探讨如何通过反射获取`Context`的实例,并展示如何利用反射实现一个集成`Toast`的工具类。这在某些场景...

    通过反射获取Context实例的Demo

    4. **获取Context实例**:最后,我们可以通过反射调用`get`方法来获取Context实例。 ```java Context context = (Context) contextField.get(null); ``` 请注意,这种方法依赖于Android框架的实现细节,可能在...

    通过反射拿到Context Demo

    在Android中,我们可以利用反射来获取并操作类的私有成员,包括私有的Context实例。 以下是使用反射获取Context的基本步骤: 1. 获取应用上下文(Application Context):通常,我们可以在Application或者Activity...

    Android 下通过反射调用加载/卸载(mount/unmount) 外置SD卡

    然后,通过反射调用这些方法并传入相应路径,执行挂载或卸载操作。 6. **权限和安全考虑**: 进行这样的操作需要较高的系统权限,通常在系统应用或者具有`android.permission.MOUNT_UNMOUNT_FILESYSTEMS`权限的...

    android通过反射获取内置存储、SD、OTG盘的路径,适合三方apk存储查看与文件管理

    5. 为了获取硬盘空间大小,可以使用`StatFs`类,传入挂载路径,然后调用`blockCountLong()`和`blockSizeLong()`方法,计算出总空间和可用空间。 这种反射技术虽然能够解决不同设备间的兼容问题,但也存在潜在的风险...

    android java反射,通过图片名获取图片

    在Android开发中,Java反射是一种强大的工具,它允许我们在运行时检查类、接口、字段和方法的信息,并且能够在不直接调用的情况下执行类的方法。在给定的场景中,“android java反射,通过图片名获取图片”这个主题...

    Android将Activity打包成jar包供第三方调用

    在Android开发中,有时我们需要将特定的功能模块,如一个Activity,封装成独立的jar包,以便第三方应用可以方便地调用。这样的做法有利于代码重用,降低开发成本,并且能够提高软件的模块化程度。本篇文章将深入探讨...

    StorageList 通过反射获取存储器列表

    1. 获取`StorageManager`实例:首先,我们需要通过`Context`的`getSystemService()`方法获取到`StorageManager`的实例,代码如下: ```java StorageManager storageManager = (StorageManager) context....

    Android获取MAC

    然后在代码中获取`WifiManager`实例并调用`getMacAddress()`方法: ```java WifiManager wifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE); String macAddress = wifiManager....

    Android 通过反射启动未安装的APK中的Activity

    在Android系统中,反射机制是一种强大的编程技巧,它允许运行时动态访问类、接口、字段和方法,即使这些元素在编译时未知。通过反射,我们可以执行一些在常规编程中无法实现的操作,例如启动未安装的APK中的Activity...

    android listview 自反射 adapter

    反射是Java语言中一个强大的特性,允许程序在运行时检查类、接口、字段和方法的信息,并可以动态调用方法或改变对象状态。 首先,我们需要了解Adapter在Android中的作用。Adapter是连接ListView和数据源的桥梁,它...

    android 通过反射自动拨打电话和挂断电话

    通过反射,我们可以在运行时动态地获取类的信息(如类名、方法名)并调用这些方法。 在Android中,拨打电话通常会使用`Intent`配合`ACTION_CALL`或`ACTION_DIAL`动作来实现。但若要通过反射实现,我们需要找到拨打...

    android 判断gps是否开的方法

    这是最简单也是最推荐的一种方法,它利用了`LocationManager`类提供的公共API来直接获取位置提供者的启用状态。 1. **获取`LocationManager`实例**: - 通过`getSystemService()`方法获取`LocationManager`实例。 ...

    Android 插件开发 实例

    由于Service涉及到系统服务的注册和生命周期管理,因此在插件化过程中需要处理更多细节,例如通过反射调用startService和stopService方法,以及适当地绑定和解绑Service。 在实际开发中,为了保证插件安全性和稳定...

    【Android】获取设备电话号码

    然后,我们可以调用`getLine1Number()`方法来获取SIM卡上的电话号码: ```java String phoneNumber = telephonyManager.getLine1Number(); ``` 需要注意的是,`getLine1Number()`方法并不总是能返回有效的电话号码...

    android7.1打开wifi热点 Demo

    在这个示例中,我们首先通过`getSystemService`获取`ConnectivityManager`,然后通过反射调用`startTethering`方法,传入参数表示我们要开启Wi-Fi热点。 此外,描述中还提到了一个Demo APK和导入方法,这通常是为了...

    Android-Cellular-Network-Info-Sample-Code-master.zip

    通过调用`getSystemService(Context.CONNECTIVITY_SERVICE)`方法可以获取到`ConnectivityManager`的实例。 在获取了`ConnectivityManager`之后,我们可以使用`getAllNetworkInfo()`方法来获取所有网络连接的信息,...

    Android代码-安卓设备信息

    通过`Context.getSystemService(Context.BATTERY_SERVICE)`获取`BatteryManager`实例,然后调用`getIntProperty(BatteryManager.BATTERY_PROPERTY_CAPACITY)`获取电池容量。 6. **传感器信息**: - **...

    Android 实现来电拦截

    首先获取`ITelephony`的实例,然后调用`endCall()`: ```java Class&lt;?&gt; telephonyClass = Class.forName("com.android.internal.telephony.ITelephony"); ITelephony telephonyService = (ITelephony) ...

Global site tag (gtag.js) - Google Analytics