前言
  
Android动态加载——加载已安装APK中的类和资源。

不错的帖子哦!

Gridview用法大总结(牛年珍藏版)+源码
http://www.eoeandroid.com/thread-190769-1-1.html

Android朴素UI城市天气预报源码
http://www.eoeandroid.com/thread-187228-1-1.html

精美Android UI界面源码(有图有真相)
http://www.eoeandroid.com/thread-187010-1-1.html

正文
一、目标

  注意:被调用的APK在Android系统中是已经安装的。
 从当前APK中调用另外一个已安装APK的字符串、颜色值、图片、布局文件资源以及Activity。

Android动态加载——加载已安装APK中的类

二、实现

  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;     } 

更多相关文章

  1. androidSDK下的图片资源
  2. 麦子学院Android应用开发工程师视频教程 共18章
  3. 使用Eclipse开发Android时整个工程或第三方jar包的正确使用
  4. Android Studio 工程视图&项目构成
  5. Android应用开发工程师项目实战 共18章(麦子学院)
  6. android 工程库及引用
  7. android:关于主工程和library project
  8. 在eclipse的android工程里引用android sdk之外的类和方法
  9. android开发资源合集

随机推荐

  1. 细解php函数
  2. 论软件测试工程师面试套路和暗语灵魂解密
  3. php学习笔记(函数相关_作用域_闭包_回调_
  4. 示例演示:数据类型的转换和检测技术、变
  5. volatile关键字
  6. string类型转Float类型
  7. 数组的排序, 数组的合并, 数组成员的统计
  8. 函数基础语法,函数进阶
  9. php常用数组函数以及数组的交集,差集,并集
  10. 数据类型的转换、变量与常量声明和使用