Android 自定义ClassLoader加载插件出现的cannot be cast to异常问题

问题定义:

Android 使用自定义ClassLoader加载插件的时候,当插件资源中使用自定义View后,会出现异常

java.lang.ClassCastException: android.support.v4.widget.SwipeRefreshLayout cannot be cast to android.support.v4.widget.SwipeRefreshLayout

</br>

问题分析:

同一个类的强制转换异常,肯定是由于不同ClassLoader加载出现的问题,打印一下两个ClassLoader,发现由布局文件解析出来的SwipeRefreshLayout是由PathClassLoader来加载,而不是由我们自定的的ClassLoader加载,应该是解析XML的时候出现的问题,查看源码中xml加载到View过程进行分析。
从Activity的setContentView()方法进入

public void setContentView(@LayoutRes int layoutResID) {
     getWindow().setContentView(layoutResID);
     initWindowDecorActionBar();
}

getWindow的mWindow对象是 com.android.internal.policy.PhoneWindow,查看PhoneWindow的setContentView()方法

public void setContentView(int layoutResID) {
        ......
        if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
            final Scene newScene = Scene.getSceneForLayout(mContentParent, layoutResID,
                    getContext());
            transitionTo(newScene);
        } else {
            mLayoutInflater.inflate(layoutResID, mContentParent);
        }
        ......
    }

XML的解析是使用mLayoutInflater的inflate方法来创建View,一直往下找,发现自定义的View是由createView()方法创建,查看该方法

 View createViewFromTag(View parent, String name, Context context, AttributeSet attrs,
            boolean ignoreThemeAttr) {
                try {
                    if (-1 == name.indexOf('.')) {
                        view = onCreateView(parent, name, attrs); //补全 android.view. ,最终还是指向createView
                    } else {
                        view = createView(name, null, attrs);
                    }
                } finally {
                    mConstructorArgs[0] = lastContext;
                }
}

private static final HashMap<String, Constructor<? extends View>> sConstructorMap =
            new HashMap<String, Constructor<? extends View>>();
public final View createView(String name, String prefix, AttributeSet attrs)
            throws ClassNotFoundException, InflateException {
        Constructor<? extends View> constructor = sConstructorMap.get(name);
        try {
            if (constructor == null) {
                //使用ClassLoader进行加载
                clazz = mContext.getClassLoader().loadClass(
                        prefix != null ? (prefix + name) : name).asSubclass(View.class);
                constructor = clazz.getConstructor(mConstructorSignature);
                constructor.setAccessible(true);
                //添加到静态缓存
                sConstructorMap.put(name, constructor);
            } else {  
            } 
            final View view = constructor.newInstance(args);
            return view;
        }  
    }

发现每个View在创建之前,都会从静态sConstructorMap中获取以前创建添加的缓存。
那么如果插件中使用到了宿主里面已经创建过的自定义View,那么进行类型转化的时候就会出现 cannot be cast to 的异常。
</br>

解决方案:

这个问题可以看成 在同一个进程中,同一个自定义View对象,是无法用不同ClassLoader加载的;
那么,不同的进程对静态变量是不共享的,所以可以对插件的活动指定不同的进程。通过对中间Activity和Service指定process来解决该问题。

android:process=":PluginProcess"

</br>

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 173,925评论 25 709
  • 在问题的最右处对问题进行等级评定,层层深入 1.Android线程之间的异步通信有哪些?(C) handler.s...
    super_shanks阅读 755评论 0 3
  • 介绍自己负责的部分,如何实现的。 自定义view viewGroup activity的启动流程 事件传递及滑动冲...
    东经315度阅读 1,258评论 1 4
  • 孩子的脚烫伤了一个多星期,这一个多星期都是的背着上学 ,今天去辅导班接他,背起他,,一百斤的妈,背起七十多斤的孩,...
    映卿阅读 154评论 0 0
  • 题目: Given a positive integer n, find the least number of ...
    八刀一闪阅读 505评论 0 0