如何用ByteBuddy替换Android活动



我正在尝试开发一个支持动态活动的Android应用程序。

由于所有活动都需要在Manifest XML中声明,所以我创建了一个什么都不做的"伪"活动。

<activity
android:name="com.research.Dummy"
android:label="@string/title_activity_dummy"
android:theme="@style/AppTheme.NoActionBar" />

我希望实现的是,使用ByteBuddy,我实例化了一个新版本的Dummy,它扩展了不同的超类和/或实现了一个或多个接口。

我有以下代码来测试扩展超类:-

final Class<? extends SomeClass> dynamicType = new ByteBuddy(ClassFileVersion.JAVA_V8)                    .subclass(com.pspdfkit.ui.PdfActivity.class, IMITATE_SUPER_CLASS)
.name("com.research.Dummy")
.method(ElementMatchers.named("toString"))
.intercept(FixedValue.value("Hello World!"))
.make()
.load(getClass().getClassLoader(), new AndroidClassLoadingStrategy.Wrapping(this.getDir("dexgen", Context.MODE_PRIVATE))).getLoaded();
final Intent intent = new Intent(this, dynamicType);
startActivity(intent);

我的Dummy活动扩展了android.support.v7.app.AppCompatActivity,但是,我希望我的ByteBuddy版本扩展"SomeClass"。

上面的代码没有给我想要的效果,Dummy活动确实显示了,但它扩展了android.support.v7.app.AppCompatActivity,而不是SomeClass。

ByteBuddy是否可以用扩展SomeClass的所需版本替换扩展AppCompatActivity的原始Dummy类?

更新

当我尝试注入我的新类时:-

final File jarFile = new File(getFilesDir(), "buddyDummy.jar");
final DynamicType.Unloaded<? extends AppCompatActivity> dynamicType = new ByteBuddy()
.subclass(AppCompatActivity.class)
.name("com.research.buddy.Dynamic")
.make();
final File dexInternalStoragePath = dynamicType.toJar(jarFile);
// Internal storage where the DexClassLoader writes the optimized dex file to.
final File optimizedDexOutputPath = getDir("outdex", Context.MODE_PRIVATE);
// Initialize the class loader with the secondary dex file.
final DexClassLoader dexClassLoader = new DexClassLoader(dexInternalStoragePath.getAbsolutePath(), optimizedDexOutputPath.getAbsolutePath(),null, getClassLoader());
dexClassLoader.loadClass("com.research.buddy.Dynamic");

我收到这个异常

java.lang.ClassNotFoundException: Didn't find class "com.research.buddy.Dynamic" on path: 
DexPathList[[zip file "/data/user/0/com.research.buddy/files/buddyDummy.jar"],nativeLibraryDirectories=[/vendor/lib, /system/lib]]
at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:56)
at java.lang.ClassLoader.loadClass(ClassLoader.java:511)
at java.lang.ClassLoader.loadClass(ClassLoader.java:469)
at com.research.buddy.Buddy.findRuntimeDependencies(Buddy.java:117)
at com.research.buddy.Buddy.onClick(Buddy.java:78)
at android.view.View.performClick(View.java:5204)
at android.view.View$PerformClick.run(View.java:21153)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:148)
at android.app.ActivityThread.main(ActivityThread.java:5417)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
Suppressed: java.io.IOException: No original dex files found for dex location /data/user/0/com.research.buddy/files/buddyDummy.jar
at dalvik.system.DexFile.openDexFileNative(Native Method)
at dalvik.system.DexFile.openDexFile(DexFile.java:295)
at dalvik.system.DexFile.<init>(DexFile.java:111)
at dalvik.system.DexFile.loadDex(DexFile.java:151)
at dalvik.system.DexPathList.loadDexFile(DexPathList.java:282)
at dalvik.system.DexPathList.makePathElements(DexPathList.java:248)
at dalvik.system.DexPathList.<init>(DexPathList.java:120)
at dalvik.system.BaseDexClassLoader.<init>(BaseDexClassLoader.java:48)
at dalvik.system.DexClassLoader.<init>(DexClassLoader.java:57)
at com.research.buddy.Buddy.findRuntimeDependencies(Buddy.java:116)
... 10 more
Suppressed: java.lang.ClassNotFoundException: Didn't find class "com.research.buddy.Dynamic" on path: 
DexPathList[[zip file "/data/app/com.research.buddy-1/base.apk"],nativeLibraryDirectories=[/data/app/com.research.buddy-1/lib/arm, /vendor/lib, /system/lib]]
at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:56)
at java.lang.ClassLoader.loadClass(ClassLoader.java:511)
at java.lang.ClassLoader.loadClass(ClassLoader.java:504)
... 12 more
Suppressed: java.lang.ClassNotFoundException: com.research.buddy.Dynamic
at java.lang.Class.classForName(Native Method)
at java.lang.BootClassLoader.findClass(ClassLoader.java:781)
at java.lang.BootClassLoader.loadClass(ClassLoader.java:841)
at java.lang.ClassLoader.loadClass(ClassLoader.java:504)
... 13 more
Caused by: java.lang.NoClassDefFoundError: Class not found using the boot class loader; no stack trace available

当我查看设备上的文件位置时,我可以在正确的位置看到buddyDummy.jar文件。

为什么我要获得ClassNotFoundException

更新

我已经成功地用ByteBuddy实例化了我的Android活动,并用以下代码"启动活动"。

final DynamicType.Unloaded<? extends AppCompatActivity> dynamicType = new ByteBuddy()
.subclass(AppCompatActivity.class)
.name(CLASS_NAME)
.make();
final Class<? extends AppCompatActivity> dynamicTypeClass = dynamicType.load(getClassLoader(), new AndroidClassLoadingStrategy.Injecting(this.getDir("dexgen", Context.MODE_PRIVATE))).getLoaded();
final Intent intent = new Intent(this, dynamicTypeClass);
startActivity(intent);

我现在需要截取onCreate方法,以便设置content

这个步骤似乎有很多问题,即onCreate是一个受保护的方法,我需要调用传递Bundle的超级类onCreate方法。

是否可以使用ByteBuddy拦截受保护的方法?如何在方法拦截器中调用super.onCreate(savedInstance)?

@Super、@SuperCall等只有call(),例如没有args,我缺少什么?

更新

我已经成功地显示了我的动态活动,设置了所需的布局,并调用了super.onCreate()。尽管我认为超级调用顺序不对。

final DynamicType.Unloaded<? extends AppCompatActivity> dynamicType = new ByteBuddy(ClassFileVersion.JAVA_V8)
.subclass(AppCompatActivity.class)
.name(CLASS_NAME)
.method(named("onCreate").and(takesArguments(1)))
.intercept(MethodDelegation.to(TargetActivity.class).andThen(SuperMethodCall.INSTANCE))
.make();
final Class<? extends AppCompatActivity> dynamicTypeClass = dynamicType.load(getClassLoader(), new AndroidClassLoadingStrategy.Injecting(this.getDir("dexgen", Context.MODE_PRIVATE))).getLoaded();
final Intent intent = new Intent(this, dynamicTypeClass);
startActivity(intent);

我的TargetActivity类似于:-

public class TargetActivity {
public static void intercept(Bundle savedInstanceState, @This AppCompatActivity thiz) {
thiz.setContentView(R.layout.activity_fourth);
}
}

由于我对SuperMethod的MethodDelegation调用是".andThen",这听起来像是在我设置内容后调用super.onCreate(),我如何调用Super.previous而不是.andThen

您使用的是Wrapping策略,其中类被加载到一个新的类加载器中。如果你想加载一个类而不是另一个类,你必须注入它

然而,这种模式可能会很尴尬,因为您假设在加载原始类之前注入该类,这通常取决于执行JVM的细节。

至于你的问题更新:你指定了正确的父类加载器吗?尝试com.research.buddy.Dynamic.class.getClassLoader()而不是getClass().getClassLoader()。您似乎将类加载到了错误的范围中。

最新更新