美文网首页
Android 自定义ClassLoader加载插件出现的can

Android 自定义ClassLoader加载插件出现的can

作者: 阿里懒 | 来源:发表于2017-05-21 21:56 被阅读0次
    问题定义:

    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 自定义ClassLoader加载插件出现的can

          本文链接:https://www.haomeiwen.com/subject/kxuvxxtx.html