【发布时间】: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