当前位置: 代码迷 >> Android >> 使用DexClassLoader与动态加载的库进行交互
  详细解决方案

使用DexClassLoader与动态加载的库进行交互

热度:94   发布时间:2023-08-04 10:16:34.0

我正在使用DexClassLoader类从SD卡在运行时加载jar文件

  final String libPath = Environment.getExternalStorageDirectory() +     "/test.jar";
        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("org.shlublu.android.sandbox.MyClass");

        final Object myInstance  = classToLoad.newInstance();
        final Method doSomething = classToLoad.getMethod("doSomething");



        doSomething.invoke(myInstance);

在我的jar文件中,我正在打印一些运行良好的日志。现在我要执行的操作是从jar文件中打印Toast。 在执行此操作时,我得到的异常是java.lang.reflect.InvocationTargetException。 我知道为什么我们得到这个异常,其背后的原因是在打印吐司时使用它的上下文中的nullpointer异常。 所以它不是重复的

原因是

  java.lang.NullPointerException: Attempt to invoke virtual method      'android.content.Context android.content.Context.getApplicationContext()' on a null object reference 

jar文件中的代码是

public class MyClass extends Activity {


@Override
protected void onCreate(Bundle savedInstanceState) {
    // TODO Auto-generated method stub
    super.onCreate(savedInstanceState);
}


public void doSomething() {

    Toast.makeText(getApplicationContext(), "MyClass: doSomething() called.", Toast.LENGTH_LONG).show();

    Log.e(MyClass.class.getName(), "MyClass: doSomething() called.");
}}

任何人都可以帮助我实现这一目标。任何帮助将不胜感激。

编辑:我想做的是我有我自己的一个库,许多客户正在使用...我想要的是从用户的SD卡中加载我的库..我想在任何时候更新该库我想要在没有用户知识的情况下也没有任何版本更新。库包含少量接口,片段和活动。 所以现在我能够从sd卡加载我的库并可以调用基本功能。现在的主要挑战是从库中实现接口并调用在其中使用了上下文的函数。

涉及此类操作的任何示例或提示将大有帮助。

我看到两种解决方法,具体取决于您愿意做什么:

  • 如果MyClass必须是一个Activity

必须使用startActivity()或任何变体来正确启动一个Activity 它也必须在清单中声明。 因此,只有当您所有的MyClass变体都具有相同的签名时,以下才起作用。

我假设您的启动代码位于Activity 在加载classToLoad ,您可以执行以下操作:

final File tmpDir = getDir("dex", 0);
final String libPath = Environment.getExternalStorageDirectory() + "/test.jar";
final DexClassLoader classloader = new DexClassLoader(libPath, tmpDir.getAbsolutePath(), null, this.getClass().getClassLoader());

try {
    final Class<Object> classToLoad = (Class<Object>) classloader.loadClass("org.shlublu.android.sandbox.MyClass");

    // CHANGED: THIS STARTS PROPERLY YOUR ACTIVITY ONCE THE CLASS LOADED
    final Intent intent = new Intent(this, classToLoad);
    startActivity(intent); 

} catch (ClassNotFoundException e) {
    // handle that Exception properly here
}

现在以某种方式更改doSomething() ,使其使用新Activity的基础Context而不是getApplicationContext() 然后例如从MyClass.onCreate()调用它,然后看它是否有效:

public class MyClass extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        doSomething(); // CHANGED: just as an example
    }

    private void doSomething() {
        // CHANGED: now the following line uses 'this' instead of `getApplicationContext()`
        Toast.makeText(this, "MyClass: doSomething() called.", Toast.LENGTH_LONG).show(); 

        Log.d(MyClass.class.getName(), "MyClass: doSomething() called.");
    }
}

其余的-什么时候调用doSomething() ,为什么等等...-完全取决于您愿意做什么。

  • 如果MyClass只需要显示一个Toast

在这种情况下,无需创建另一个Activity doSomething()仅需要接收适当的Context即可显示Toast

如下更改MyClass

public class MyClass {
    private void doSomething(Context ctx) {
        Toast.makeText(ctx, "MyClass: doSomething() called.", Toast.LENGTH_LONG).show();

        Log.d(MyClass.class.getName(), "MyClass: doSomething() called.");
    }
}

改变你的启动代码通过thisdoSomething()假设它是从运行的Activity

final File tmpDir = getDir("dex", 0);
final String libPath = Environment.getExternalStorageDirectory() + "/test.jar";
final DexClassLoader classloader = new DexClassLoader(libPath, tmpDir.getAbsolutePath(), null, this.getClass().getClassLoader());

try {
    final Class<Object> classToLoad = (Class<Object>) classloader.loadClass("org.shlublu.android.sandbox.MyClass");

    // CHANGED: LOADS THE METHOD doSomething(Context). EXECUTES IT WITH this AS AN ARGUMENT
    final Class[] args = new Class[1];
    args[0] = Context.class;
    final Method doSomething = classToLoad.getMethod("doSomething", args);

    final Object myInstance  = classToLoad.newInstance();

    doSomething.invoke(myInstance, this);    
} catch (ClassNotFoundException e) {
    // handle that Exception properly here
}