18

我正在尝试做与此stackoverflow 发布类似的事情。我想做的是从 SD 卡中读取活动或服务的定义。为避免清单权限问题,我在 .apk 中创建了此活动的 shell 版本,但尝试将其替换为运行时驻留在 SD 卡上的同名活动。不幸的是,我可以使用 DexClassLoader 从 SD 卡加载活动类定义,但执行的是原始类定义。有没有办法指定新的类定义替换旧的类定义,或者有什么建议可以避免清单权限问题而不实际在包中提供所需的活动?代码示例:

    ClassLoader cl = new DexClassLoader("/sdcard/mypath/My.apk",
            getFilesDir().getAbsolutePath(),
            null,
            MainActivity.class.getClassLoader());

    try {
        Class<?> c = cl.loadClass("com.android.my.path.to.a.loaded.activity");
        Intent i = new Intent(getBaseContext(), c);
        startActivity(i);
    }
    catch (Exception e) {

它不会启动 in 中com.android.my.path.to.a.loaded.activity指定的/sdcard/mypath/My.apk内容,而是启动静态加载到项目中的活动。

4

1 回答 1

16

通过 DexClassLoader 启动一个 Activity 会很棘手,因为你没有安装 APK,所以没有什么可以处理你的 Intent。

您的平台 APK 中应该有相同的活动类,并在 AndroidManifest.xml 中声明它。然后,您应该将当前的 ClassLoader 更改为您想要的 DexClassLoader。结果,它将启动您的插件 APK。(注:“平台APK”是指手机中已经安装的应用程序,“插件APK”是指SD卡中保存的apk文件。)

平台的应用程序应如下所示:

public static ClassLoader ORIGINAL_LOADER;
public static ClassLoader CUSTOM_LOADER = null;

@Override
public void onCreate() {
    super.onCreate();
    try {
        Context mBase = new Smith<Context>(this, "mBase").get();

        Object mPackageInfo = new Smith<Object>(mBase, "mPackageInfo")
                .get();
        //get Application classLoader
        Smith<ClassLoader> sClassLoader = new Smith<ClassLoader>(
                mPackageInfo, "mClassLoader");
        ClassLoader mClassLoader = sClassLoader.get();
        ORIGINAL_LOADER = mClassLoader;

        MyClassLoader cl = new MyClassLoader(mClassLoader);
        //chage current classLoader to MyClassLoader
        sClassLoader.set(cl);
    } catch (Exception e) {
        e.printStackTrace();
    }
}

class MyClassLoader extends ClassLoader {
    public MyClassLoader(ClassLoader parent) {
        super(parent);
    }

    @Override
    public Class<?> loadClass(String className)
            throws ClassNotFoundException {
        if (CUSTOM_LOADER != null) {
            try {
                Class<?> c = CUSTOM_LOADER.loadClass(className);
                if (c != null)
                    return c;
            } catch (ClassNotFoundException e) {
            }
        }
        return super.loadClass(className);
    }
}

更多代码,可以访问https://github.com/Rookery/AndroidDynamicLoader

我正在阅读 Android 源代码,以便找到一种更优雅的方法来实现此功能。如果您有任何想法,请随时与我联系。

于 2013-04-08T03:21:18.610 回答