доступ к классам приложения из файла dex с помощью загрузчика классов

У меня есть приложение, которое получает файл dex с сервера, затем сохраняет его на SD-карте и загружает для

выполнение некоторых функций. Я загружаю класс моего приложения из моего файла dex следующим образом

dex-файл

public class Main  {


    public void onCreate() 
    {
        System.out.print("------------onCreate------------");
        try {
             final ClassLoader classloader = ClassLoader.getSystemClassLoader();
            final Class<Object> classToLoad = (Class<Object>) classloader.loadClass("com.example.myapp.M");
            final Object myInstance  = classToLoad.newInstance();
            final Method doSomething = classToLoad.getMethod("doSomething");
            doSomething.invoke(myInstance);

        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

код приложения

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        try {
            final String libPath = Environment.getExternalStorageDirectory() + "/Lib/lib.apk";
            final File tmpDir = getDir("dex", 0);
            final DexClassLoader classloader = new DexClassLoader(libPath, tmpDir.getAbsolutePath(), null, this.getClass().getClassLoader());
            final Class<Object> classToLoad = (Class<Object>) classloader.loadClass("com.example.lib.Main");
            final Object myInstance  = classToLoad.newInstance();
            final Method doSomething = classToLoad.getMethod("onCreate");
            doSomething.invoke(myInstance);
            } catch (Exception e) {
            e.printStackTrace();
        }
    }

М класс в моем приложении

public class M {
public void doSomething()
{
    Log.e("doSomething", "--------------------doSomething----");
}
}

но у меня следующая ошибка. Что не так в моем коде

11-19 08:24:12.718: W/System.err(10867): java.lang.ClassNotFoundException: Didn't find class "com.example.myapp2.M" on path: .
11-19 08:24:12.738: W/System.err(10867):    at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:65)
11-19 08:24:12.747: W/System.err(10867):    at java.lang.ClassLoader.loadClass(ClassLoader.java:501)
11-19 08:24:12.758: W/System.err(10867):    at java.lang.ClassLoader.loadClass(ClassLoader.java:461)
11-19 08:24:12.758: W/System.err(10867):    at com.example.lib.Main.onCreate(Main.java:15)
11-19 08:24:12.768: W/System.err(10867):    at java.lang.reflect.Method.invokeNative(Native Method)
11-19 08:24:12.788: W/System.err(10867):    at java.lang.reflect.Method.invoke(Method.java:511)
11-19 08:24:12.788: W/System.err(10867):    at com.example.myapp2.MainActivity.onCreate(MainActivity.java:28)
11-19 08:24:12.798: W/System.err(10867):    at android.app.Activity.performCreate(Activity.java:5104)
11-19 08:24:12.818: W/System.err(10867):    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1080)
11-19 08:24:12.828: W/System.err(10867):    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2144)
11-19 08:24:12.828: W/System.err(10867):    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2230)
11-19 08:24:12.828: W/System.err(10867):    at android.app.ActivityThread.access$600(ActivityThread.java:141)
11-19 08:24:12.828: W/System.err(10867):    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1234)
11-19 08:24:12.828: W/System.err(10867):    at android.os.Handler.dispatchMessage(Handler.java:99)
11-19 08:24:12.838: W/System.err(10867):    at android.os.Looper.loop(Looper.java:137)
11-19 08:24:12.847: W/System.err(10867):    at android.app.ActivityThread.main(ActivityThread.java:5041)
11-19 08:24:12.847: W/System.err(10867):    at java.lang.reflect.Method.invokeNative(Native Method)
11-19 08:24:12.847: W/System.err(10867):    at java.lang.reflect.Method.invoke(Method.java:511)
11-19 08:24:12.857: W/System.err(10867):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793)
11-19 08:24:12.898: W/System.err(10867):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560)
11-19 08:24:12.898: W/System.err(10867):    at dalvik.system.NativeStart.main(Native Method)

person zohreh    schedule 19.11.2013    source источник
comment
Что вам непонятно в сообщении ClassNotFoundException: Didn't find class on path .   -  person Marek Sebera    schedule 19.11.2013
comment
path.can я могу получить доступ к классам моего приложения из файла dex с помощью classloader.if да, как указать путь для доступа?   -  person zohreh    schedule 19.11.2013


Ответы (3)


Проверьте, следуете ли вы этим шагам.

  1. У вас есть файл библиотеки в виде APK или JAR, содержащего файл (Classes.dex) на SD-карте?
  2. Скопируйте APK-файл во временную папку в папку вашего приложения /data/data/your.app.name/tmp. Это важно, так как из соображений безопасности Android не позволяет загружать что-либо с SD-карты.
  3. Используйте DexClassLoader, как описано здесь -> Как загрузить Java класс динамически на android/dalvik?
  4. Используйте этот экземпляр загрузчика классов для загрузки любых файлов динамических классов.

Если у вас более 1 файла JAR, объедините их в один файл DEX/APK.

Чтобы преобразовать ваш JAR-файл или классы Java в DEX, вам нужно использовать dx.bat Android SDK. Вот пример сценария ANT. Это также можно сделать через командную строку с аналогичными аргументами.

<exec executable="dx.bat">
    <arg value="--dex" />
    <arg value="--output=${target.directory.fullpath}/${apk.name}" />
    <arg value="--positions=lines" />
    <arg path="${classes.directory.full.path}" />
</exec>
person Swaroop    schedule 19.11.2013
comment
Могу ли я получить доступ к классам моего приложения из моего файла jar? Можете ли вы объяснить, как я могу это сделать? - person zohreh; 19.11.2013
comment
Я хочу, чтобы класс моего файла jar вызывал метод моего приложения - person zohreh; 19.11.2013
comment
Да, ваши подключаемые файлы JAR (формат DEX) могут вызывать методы вашего приложения без каких-либо явных действий, поскольку они должны действовать как загрузчик родительского класса. - person Swaroop; 19.11.2013
comment
См. конструктор для DexClassLoader, где вы устанавливаете загрузчик родительского класса в качестве одного из входных данных. DexClassLoader. Вы можете получить загрузчик родительского класса, используя метод activity.getClassLoader() для перехода в конструктор DexClassLoader() - person Swaroop; 19.11.2013
comment
я использовал приведенный выше код в файле jar, и я снова получаю эту ошибку. Что такое путь в ошибке? - person zohreh; 19.11.2013

Я нашел свой ответ, мне нужно изменить

final Class<Object> classToLoad = (Class<Object>)classloader.loadClass("com.example.myapp.M");

to

final Class<Object> classToLoad = (Class<Object>)Class.forName("com.example.myapp.M");

в моем файле dex

person zohreh    schedule 19.11.2013
comment
странно, как это сработало для вас. Вы нигде не упоминаете путь к файлу .jar... - person Harry Sharma; 16.10.2015
comment
Итак, если вы сделаете это, загрузит ли он все остальные классы, которые нужны com.example.myapp.M (которые еще не загружены)? - person LarsH; 01.08.2016
comment
Zohreh, вы загружали файлы классов с SD-карты? Или какой-то другой каталог внешнего хранилища? Мне любопытно, потому что @Swaroop сказал, что вы не можете загрузить их с SD-карты. - person LarsH; 27.08.2016

@zohreh Замените строку ниже

final DexClassLoader classloader = new DexClassLoader(libPath, tmpDir.getAbsolutePath(), null, this.getClass().getClassLoader());

to

final DexClassLoader classloader = new DexClassLoader(libPath, tmpDir.getAbsolutePath(), null, ClassLoader.getSystemClassLoader());
person Ashish Verma    schedule 24.05.2017