由意外的DEX解决的类;

时间:2014-03-05 09:32:18

标签: android reflection dex dexclassloader

我正在开发一个应用程序,它从外部apk文件自动加载来自外部dex的类(外部apk文件存储在应用程序的内部存储中)。外部apk文件使用com.google.gson进行分类。

应用程序的源代码

// Internal storage where the DexClassLoader writes the optimized dex file to.
    final File optimizedDexOutputPath = mContext.getDir("optimize_offload", Context.MODE_PRIVATE);

    // Initialize the class loader with the s dex file.
    DexClassLoader dexClassLoader = new DexClassLoader(apkPath,  optimizedDexOutputPath.getAbsolutePath(),
            null,  mContext.getClassLoader());

    // Load the apk class from the class loader.
    Class<?> apkProviderClass = dexClassLoader.loadClass(className);        

    Object obj = apkProviderClass.newInstance();   

    // call method wrappers
    java.lang.reflect.Method method = apkProviderClass.getDeclaredMethod("R_" + methodName, String.class);
    System.out.println(method.getName());

    method.setAccessible(true); 

    Object result = method.invoke(obj, params); // Exception here

外部apk文件中的类具有以下方法

public long R_catalanSum(String params) {

    Gson gson = new Gson();

    StartedSumLooperService_catalanSum param = 
            gson.fromJson(params, StartedSumLooperService_catalanSum.class);

    // Call local implementation
    return catalanSum(param.n); 
}

public long R_fiboSum(String params) {

    Gson gson = new Gson();

    StartedSumLooperService_fiboSum param = 
            gson.fromJson(params, StartedSumLooperService_fiboSum.class);

    // Call local implementation
    return fiboSum(param.n); 
}

但是我在跑步时面临以下问题:

03-05 17:10:59.479: W/dalvikvm(5433): Class resolved by unexpected DEX: Lorg/vkedco/android/startedsumlooperservice/StartedSumLooperService;(0x40f007d8):0x56f3a000 ref [Lcom/google/gson/Gson;] Lcom/google/gson/Gson;(0x40eabac0):0x56daa000

03-05 17:10:59.479:W / dalvikvm(5433):( Lorg / vkedco / android / startedsumlooperservice / StartedSumLooperService;使用了不同的Lcom / google / gson / Gson;在预验证期间)

根据Android构建过程,dex文件和第三方库(例如gson)被捆绑到dex文件(classes.dex)中。因此,在使用DexClassLoader初始化类加载器时,我不需要手动指定所有第3个库。我是否正确?

3 个答案:

答案 0 :(得分:2)

这个问题的答案是DEX在类加载器和加载类中找到2个版本的gson。 解决方案是:只在类加载器或加载类中使用gson。

答案 1 :(得分:1)

在这个https://groups.google.com/forum/#!topic/android-platform/IP9BuSXgTdQ的谷歌论坛上似乎有一个问题的解决方案。

答案 2 :(得分:0)

我遇到了同样的问题。

但现在已经解决了。我们可以在loader和loading类中使用lib。 我用eclipse。

在loader和loading项目中,将jsoup.jar添加到libs文件夹和lib引用

enter image description here

enter image description here

然后我们可以在类加载器和加载中使用相同的lib。