您的位置:首页 > 移动开发 > Android开发

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

2016-09-29 22:03 751 查看
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代码



package
chroya.demo;

import
android.app.Activity;

import
android.os.Bundle;

import
android.util.Log;

class
Main extends
Activity {

@Override

publicvoid
onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

}

publicvoid
print(String msg) {

Log.d("Main","msg:"+
msg);

}

}

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

Java代码



Context
c = createPackageContext("chroya.demo",
Context.CONTEXT_INCLUDE_CODE | Context.CONTEXT_IGNORE_SECURITY);

//载入这个类

Class clazz = c.getClassLoader().loadClass("chroya.demo.Main");

//新建一个实例

Object owner = clazz.newInstance();

//获取print方法,传入参数并执行

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代码

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了。

在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);
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  android