【问题标题】:Android : load(getClass().getClassLoader(),classLoadingStrategy) throws java.lang.IllegalStateExceptionAndroid : load(getClass().getClassLoader(),classLoadingStrategy) 抛出 java.lang.IllegalStateException
【发布时间】:2015-06-11 21:13:00
【问题描述】:

我试图在 Android 中的方法 proxy(...) 下创建 byteBuddy 对象,然后在 byteBuddy 对象上调用一些方法:

<T> T proxy(Class<T> clz, InvocationHandler invocationHandler) {
    ByteBuddy byteBuddy = null;
    try {
        byteBuddy = new ByteBuddy(ClassFileVersion.JAVA_V6);
    } catch (Throwable e) {
        //e.printStackTrace();
    }
    if(byteBuddy!=null) {
        Class<?> enhanced = byteBuddy
                .subclass(clz, ConstructorStrategy.Default.IMITATE_SUPER_TYPE)
                .method(ElementMatchers.not(ElementMatchers.isDeclaredBy(Object.class)))
                .intercept(InvocationHandlerAdapter.of(invocationHandler))
                .make().load(getClass().getClassLoader(), classLoadingStrategy)
                .getLoaded();

        ...

但符合:

load(getClass().getClassLoader(), classLoadingStrategy)

ByteBuddy 抛出异常:

06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:java.lang.IllegalStateException:无法加载类 pdl.transport.overlay.fissione.FissioneTransport$FissoneHandler$ByteBuddy$vhLwLk79 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 net.bytebuddy.android.AndroidClassLoadingStrategy.load(AndroidClassLoadingStrategy.java:138) 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 net.bytebuddy.dynamic.DynamicType$Default$Unloaded.load(DynamicType.java:3380) 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 pdl.transport.Messenger.proxy(Messenger.java:320) 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 pdl.transport.Messenger.async(Messenger.java:382) 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 pdl.transport.Messenger.async(Messenger.java:373) 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 pdl.transport.overlay.fissione.FissioneTransport.join(FissioneTransport.java:221) 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 pdl.transport.overlay.fissione.FissioneTransport.open(FissioneTransport.java:202) 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 pdl.transport.overlay.util.DHT.(DHT.java:37) 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 pdl.transport.overlay.util.DHT.main(DHT.java:117) 06-11 21:29:23.351 12028-12028/com.example.AndroidOverlay W/System.err:在 com.example.AndroidOverlay.MyActivity_newbie.onCreate(MyActivity_newbie.java:72) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 android.app.Activity.performCreate(Activity.java:6289) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err﹕在 android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1119) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2646) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2758) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 android.app.ActivityThread.access$900(ActivityThread.java:177) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err﹕在 android.app.ActivityThread$H.handleMessage(ActivityThread.java:1448) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 android.os.Handler.dispatchMessage(Handler.java:102) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 android.os.Looper.loop(Looper.java:145) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err﹕在 android.app.ActivityThread.main(ActivityThread.java:5942) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err﹕在 java.lang.reflect.Method.invoke(本机方法)06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 java.lang.reflect.Method.invoke(Method.java:372) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1400) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1195) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err﹕ 引起:java.lang.ClassNotFoundException:找不到类 “pdl.transport.overlay.fissione.FissioneTransport$FissoneHandler$ByteBuddy$vhLwLk79” 在路径上:DexPathList[[zip 文件 "/data/data/com.example.AndroidOverlay/app_TnUR5LUb/1Gzh1FCI.jar"],nativeLibraryDirectories=[/vendor/lib, /system/lib]] 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:56) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 java.lang.ClassLoader.loadClass(ClassLoader.java:511) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 java.lang.ClassLoader.loadClass(ClassLoader.java:469) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 net.bytebuddy.android.AndroidClassLoadingStrategy.load(AndroidClassLoadingStrategy.java:136) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err: ... 22 更多 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:被抑制: java.io.IOException:无法打开 dex 文件 '/data/data/com.example.AndroidOverlay/app_TnUR5LUb/1Gzh1FCI.jar' 来自 内存:无法识别的版本号 /data/data/com.example.AndroidOverlay/app_TnUR5LUb/1Gzh1FCI.jar:0 3 6 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 dalvik.system.DexFile.openDexFileNative(Native 方法)06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 dalvik.system.DexFile.openDexFile(DexFile.java:295) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 dalvik.system.DexFile.(DexFile.java:111) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err﹕在 dalvik.system.DexFile.loadDex(DexFile.java:151) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 dalvik.system.DexPathList.loadDexFile(DexPathList.java:265) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err﹕在 dalvik.system.DexPathList.makeDexElements(DexPathList.java:231) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err﹕在 dalvik.system.DexPathList.(DexPathList.java:109) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err﹕在 dalvik.system.BaseDexClassLoader.(BaseDexClassLoader.java:48) 06-11 21:29:23.361 12028-12028/com.example.AndroidOverlay W/System.err:在 dalvik.system.DexClassLoader.(DexClassLoader.java:57) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err﹕在 net.bytebuddy.android.AndroidClassLoadingStrategy$DexProcessor$ForSdkCompiler.makeClassLoader(AndroidClassLoadingStrategy.java:257) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err:在 net.bytebuddy.android.AndroidClassLoadingStrategy.load(AndroidClassLoadingStrategy.java:132) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err: ... 22 更多 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err:由以下原因引起: java.io.IOException:执行失败(/system/bin/dex2oat --runtime-arg -classpath --runtime-arg --instruction-set=arm --instruction-set-features=div --runtime-arg -Xrelocate --boot-image=/system/framework/boot.art --dex-file= /data/data/com.example.AndroidOverlay/app_TnUR5LUb/1Gzh1FCI.jar --oat-fd=64 --art-fd=-1 --oat-location=/data/data/com.example.AndroidOverlay/app_TnUR5LUb/1Gzh1FCI.dex --runtime-arg -Xms64m --runtime-arg -Xmx512m) 因为非 0 退出状态 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err: ... 33 更多 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err:由以下原因引起: java.io.IOException: 找不到 dex 文件 '/data/data/com.example.AndroidOverlay/app_TnUR5LUb/1Gzh1FCI.jar' 在 燕麦位置 '/data/data/com.example.AndroidOverlay/app_TnUR5LUb/1Gzh1FCI.dex': 无法在以下位置找到现有的 oat 文件 /data/data/com.example.AndroidOverlay/app_TnUR5LUb/1Gzh1FCI.dex:文件 0 字节的大小不足以包含 52 字节的 ELF 标头: '/data/data/com.example.AndroidOverlay/app_TnUR5LUb/1Gzh1FCI.dex' 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err: ... 33 更多 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err:被抑制: java.lang.ClassNotFoundException:找不到类 “pdl.transport.overlay.fissione.FissioneTransport$FissoneHandler$ByteBuddy$vhLwLk79” 在路径上:DexPathList[[zip 文件 "/data/app/com.example.AndroidOverlay-1/base.apk"],nativeLibraryDirectories=[/vendor/lib, /system/lib]] 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err:在 dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:56) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err:在 java.lang.ClassLoader.loadClass(ClassLoader.java:511) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err:在 java.lang.ClassLoader.loadClass(ClassLoader.java:504) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err: ... 24 更多 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err:被抑制: java.lang.ClassNotFoundException: pdl.transport.overlay.fissione.FissioneTransport$FissoneHandler$ByteBuddy$vhLwLk79 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err:在 java.lang.Class.classForName(Native Method) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err﹕在 java.lang.BootClassLoader.findClass(ClassLoader.java:781) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err﹕在 java.lang.BootClassLoader.loadClass(ClassLoader.java:841) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err﹕在 java.lang.ClassLoader.loadClass(ClassLoader.java:504) 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err: ... 25 更多 06-11 21:29:23.371 12028-12028/com.example.AndroidOverlay W/System.err﹕由:java.lang.NoClassDefFoundError: Class not 使用引导类加载器找到;没有可用的堆栈

我正在使用来自http://central.maven.org/maven2/com/google/android/tools/dx/1.7/dx-1.7.jar 的 dx-1.7.jar 和 byte-buddy-0.6.8.jar,byte-buddy-android-0.6.8.jar

我错过了什么吗?我也无法在您的 android 示例应用程序上运行动态类型。(解析失败:Lorg/objectweb/asmType;),好像我错过了一些东西,一些库或者我不知道...... 谢谢你的回答。

EDIT1: 我使用 Lollipop,这就是我设置 AndroidClassLoadingStrategy 的方式:

final File dir = this.getDir("dexgen", Context.MODE_PRIVATE);
    Messenger.setClassLoadingStrategy(new AndroidClassLoadingStrategy(dir));

在我有方法 proxy(...) 的地方设置为类 Messenger

public static void setClassLoadingStrategy(ClassLoadingStrategy cls) {
    classLoadingStrategy = cls;
}

classLoadingStrategy 被定义为

private static ClassLoadingStrategy classLoadingStrategy;

编辑 2: 毕竟我拿了 ByteBuddy 示例应用程序,尝试了 Android lollipop 替换的行

File file = TestActivity.this.getDir(RandomString.make(), Context.MODE_PRIVATE);

对于应该替换的其中一个,尝试了所有这些并且都给出了相同的异常,只是来自不同的类活动

File file = getCodeCacheDir(); //NOT WORKING
// File file = getApplicationContext().getCodeCacheDir(); //NOT WORKING
// File file = getBaseContext().getCodeCacheDir(); //NOT WORKING
// File file = TestActivity.this.getCodeCacheDir(); //NOT WORKING

异常仍然:

06-12 23:18:57.916    1947-1947/net.bytebuddy.android.test W/net.bytebuddy﹕ java.lang.IllegalStateException: Cannot load class net.bytebuddy.renamed.java.lang.Object$ByteBuddy$uSYJ5787$auxiliary$MBywjCuh

我创建并放置了我的网站资源,以便将来在此处重现错误:

bashism.com/shared/ByteBuddyTest.tar.gz

编辑 3:

Android 版本:5.0.1 Lollipop

设备:三星 Galaxy S4

IDE:IntelliJ Idea 14.1.3

编辑 4: 在 Main Activity 内部更改后:

File file = TestActivity.this.getDir(RandomString.make(), Context.MODE_PRIVATE);

在 net.bytebuddy.android.AndroidClassLoadingStrategy 内部

public ForSdkCompiler(DexOptions dexFileOptions, CfOptions dexCompilerOptions) {
dexFileOptions.targetApiLevel = 13;
this.dexFileOptions = dexFileOptions;
this.dexCompilerOptions = dexCompilerOptions;

}

看起来工作/稳定,很快会进一步调试

【问题讨论】:

  • 您使用的是哪个安卓版本?你使用什么类加载策略?它必须是AndroidClassLoadingStrategy。你如何设置它?您需要使用示例应用程序中的内部文件夹,否则 Android 的安全管理器会启动。
  • 我刚刚添加了更多信息的编辑,谢谢
  • 看来你需要使用File dir = context.getCodeCacheDir() 在 Lollipop 上存储 dex 文件。 (developer.android.com/reference/dalvik/system/…) 用于存储文件。不幸的是,Android 用户倾向于更改他们的安全设置和 API。
  • 你好,对不起,我无法在applicationContext或Activity下找到getCodeCacheDir方法,怎么从activity调用呢?谢谢你,麻烦你了……啊,我明白了,我的目标设置为不同,所以我无法调用它
  • 您需要针对 API 版本 21(即 Lollipop)进行编译。此版本在Activity 实现的ContextWrapper 子类上添加了getCodeCacheDir 方法。

标签: java android byte-buddy


【解决方案1】:

正如我在上面的评论中提到的那样,问题是生成的 DEX 文件当前太新(较新的版本显然引入了“扩展操作码”),无法处理底层 ART 机器。实际问题可能源于具有不同默认 DexOptions.targetApiLevel 值的不同设备(这可能会忽略实际的 API 构建目标)。要解决此问题,任何涉及classes.dex 文件的类加载策略都应将此值专门设置为13 或更低,以确保生成正确的dex 文件。例如,在@raphw 的AndroidClassLoadingStrategy 中,可以修改DexProcessor.ForSdkCompiler 构造函数以专门设置此值,例如:

public ForSdkCompiler(DexOptions dexFileOptions, CfOptions dexCompilerOptions) {
    dexFileOptions.targetApiLevel = 13;
    this.dexFileOptions = dexFileOptions;
    this.dexCompilerOptions = dexCompilerOptions;
}

【讨论】:

  • 您好,所以我尝试了您的修复,并且在 bytebuddy 示例中它似乎工作且稳定,稍后我会在进行更多调试和研究时将答案标记为有用。谢谢
  • 我最近拿到了一部新手机,我终于可以验证报告的错误了。另外,我可以确认您的解决方案有效。我明天将发布一个新版本的 Byte Buddy,您的修复程序将在其中进行调整。感谢您对此进行研究!
猜你喜欢
  • 2010-11-22
  • 1970-01-01
  • 1970-01-01
  • 2013-01-22
  • 2016-02-10
  • 2017-08-12
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多