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

Android应用开发提高系列(5)——Android动态加载(下)——加载已安装APK中的类和资源

2012-04-19 18:10 567 查看
声明

  欢迎转载,但请保留文章原始出处:)

    博客园:http://www.cnblogs.com

    农民伯伯: http://over140.cnblogs.com
    Android中文Wiki:http://wikidroid.sinaapp.com

正文

  一、目标

    注意被调用的APK在Android系统中是已经安装的。

    从当前APK中调用另外一个已安装APK的字符串、颜色值、图片、布局文件资源以及Activity。

    


  二、实现

    2.1   被调用工程

       基本沿用上个工程的,添加了被调用的字符串、图片等,所以这里就不贴了,后面有下载工程的链接。

    2.2   调用工程代码



public class TestAActivity extends Activity {

/** TestB包名 */

private static final String PACKAGE_TEST_B = "com.nmbb.b";

@Override

public void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.main);

try {

final Context ctxTestB = getTestBContext();

Resources res = ctxTestB.getResources();

// 获取字符串string

String hello = res.getString(getId(res, "string", "hello"));

((TextView) findViewById(R.id.testb_string)).setText(hello);

// 获取图片Drawable

Drawable drawable = res

.getDrawable(getId(res, "drawable", "testb"));

((ImageView) findViewById(R.id.testb_drawable))

.setImageDrawable(drawable);

// 获取颜色值

int color = res.getColor(getId(res, "color", "white"));

((TextView) findViewById(R.id.testb_color))

.setBackgroundColor(color);

// 获取布局文件

View view = getView(ctxTestB, getId(res, "layout", "main"));

LinearLayout layout = (LinearLayout) findViewById(R.id.testb_layout);

layout.addView(view);

// 启动TestB Activity

findViewById(R.id.testb_activity).setOnClickListener(

new OnClickListener() {

@Override

public void onClick(View v) {

try {

@SuppressWarnings("rawtypes")

Class cls = ctxTestB.getClassLoader()

.loadClass("com.nmbb.TestBActivity");

startActivity(new Intent(ctxTestB, cls));

} catch (ClassNotFoundException e) {

e.printStackTrace();

}

}

});

} catch (NameNotFoundException e) {

e.printStackTrace();

}

}

/**

* 获取资源对应的编号

*

* @param testb

* @param resName

* @param resType

* layout、drawable、string

* @return

*/

private int getId(Resources testb, String resType, String resName) {

return testb.getIdentifier(resName, resType, PACKAGE_TEST_B);

}

/**

* 获取视图

*

* @param ctx

* @param id

* @return

*/

public View getView(Context ctx, int id) {

return ((LayoutInflater) ctx

.getSystemService(Context.LAYOUT_INFLATER_SERVICE)).inflate(id,

null);

}

/**

* 获取TestB的Context

*

* @return

* @throws NameNotFoundException

*/

private Context getTestBContext() throws NameNotFoundException {

return createPackageContext(PACKAGE_TEST_B,

Context.CONTEXT_IGNORE_SECURITY | Context.CONTEXT_INCLUDE_CODE);



}

    代码说明:

      基本原理:通过package获取被调用应用的Context,通过Context获取相应的资源、类。

    注意:

      a). 网上许多文章是通过当前工程的R.id来调用被调用工程的资源 ,这是错误的,即使不报错那也是凑巧,因为R是自动生成的,两个应用的id是没有办法对应的,所以需要通过getIdentifier来查找。

      b). Context.CONTEXT_INCLUDE_CODE一般情况下是不需要加的,如果layout里面包含了自定义控件,就需要加上。注意不能在当前工程强制转换获得这个自定义控件,因为这是在两个ClassLoader中,无法转换。
      c). 获取这些资源是不需要shareUserId的。

  三、总结
    与上篇文章相比,获取资源更加方便,但也存在一些限制:

    3.1  被调用的apk必须已经安装,降低用户体验。

    3.2  style是无法动态设置的,即使能够取到。

    3.3  从目前研究结果来看,被调用工程如果使用自定义控件,会受到比较大的限制,不能强制转换使用(原因前面已经讲过)。

    3.4  由于一个工程里面混入了两个Context,比较容易造成混淆,取资源也比较麻烦。这里分享一下批量隐射两个apk id的办法,可以通过反射获取两个apk的R类,一次获取每一个id和值,通过名称一一匹配上,这样就不用手工传入字符串了。



@SuppressWarnings("rawtypes")

private static HashMap<String, Integer> getR(Class cls) throws ClassNotFoundException, InstantiationException, IllegalAccessException {

HashMap<String, Integer> result = new HashMap<String, Integer>();

for (Class r : cls.getClasses()) {

if (!r.getName().endsWith("styleable")) {

Object owner = r.newInstance();

for (Field field : r.getFields()) {

result.put(field.getName(), field.getInt(owner));

}

}

}

return result;



}

  四、下载

     Test2012-4-19.zip

  五、文章

    Android类动态加载技术

结束

  如果是做大面积的换肤,还比较复杂,这种方式也不是很方便,这也是为什么现在市面上做换肤的很少,有也是很简单的换肤。这几天想到的另外一个方案,还没有实践,有效果了再拿出来分享,欢迎大家交流 :)
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐