如何用 ByteBuddy 替换 Android Activity

How to replace an Android Activity with ByteBuddy

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

由于所有活动都需要在清单中声明 XML 我创建了一个什么都不做的 "dummy" activity。

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

我希望实现的是使用 ByteBuddy 我实例化一个新版本的 Dummy 扩展不同的 superclasses and/or 实现一个或多个接口。

我有这个代码来测试扩展超级class:-

    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 activity 扩展 android.support.v7.app.AppCompatActivity,但是,我希望我的 ByteBuddy 版本扩展“SomeClass”。

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

ByteBuddy 能否将扩展 AppCompatActivity 的原始 Dummy class 替换为扩展 SomeClass 的所需版本?

更新

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

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 Activity 并使用以下代码 "startActivity" 它。

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的超级classes onCreate方法。

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

@Super、@SuperCall 等只有 call() 例如没有参数,我错过了什么?

更新

我已成功显示我的动态 activity,设置所需的布局并调用 super.onCreate()。虽然我认为 super 调用顺序不对。

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);

我的目标Activity 类似于:-

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 策略,其中 class 被加载到新的 class 加载器中。如果你想加载一个 class 而不是另一个 class,你必须注入它。

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

关于您的问题更新:您是否指定了正确的父 class 加载程序?而不是 getClass().getClassLoader() 尝试 com.research.buddy.Dynamic.class.getClassLoader()。看来您正在将 class 加载到错误的范围。