首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >使用DexClassLoader与动态加载库交互

使用DexClassLoader与动态加载库交互
EN

Stack Overflow用户
提问于 2015-10-19 10:56:38
回答 1查看 1.4K关注 0票数 1

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

代码语言:javascript
复制
  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文件中,我打印了几个日志,这些日志正在运行fine.Now,我想要做的是从jar文件中打印一个Toast。在这样做的时候,我得到的例外是java.lang.reflect.InvocationTargetException。我知道为什么我们会得到这个异常,并且它背后的原因是在上下文中使用空指针异常,同时使用它来打印toast。所以它不是重复的

What could cause java.lang.reflect.InvocationTargetException?

因为它背后是

代码语言:javascript
复制
  java.lang.NullPointerException: Attempt to invoke virtual method      'android.content.Context android.content.Context.getApplicationContext()' on a null object reference 

jar文件中的代码是

代码语言:javascript
复制
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.");
}}

谁能帮我实现this.Any的帮助将不胜感激。

编辑:我想要做的是我有一个我自己的库,我的许多客户都是using...what,我想要从用户的SD card..and加载我的库,当我想在没有用户知识和没有任何版本的情况下更新库时,updates.Library包含很少的接口、片段和活动。因此,现在我可以从sd卡加载我的库,并且可以调用基本的functions.Now,主要的挑战是从库中实现接口,并调用函数中的上下文。

任何涉及此类操作的示例或提示都将非常有用。

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2015-10-19 12:33:40

我看到了解决这个问题的两种方法,取决于你愿意做什么:

  • 如果MyClass Activity 必须是的话

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

我假设您的启动代码位于一个Activity中。在加载了classToLoad之后,您可以执行以下操作:

代码语言:javascript
复制
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()调用它,并确保它工作:

代码语言:javascript
复制
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 只需显示,则

在这种情况下,不需要创建另一个ActivitydoSomething()只需要接收适当的Context就可以显示Toast

更改MyClass如下:

代码语言:javascript
复制
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.");
    }
}

并将启动代码更改为将this传递给doSomething(),假设它是从Activity运行的。

代码语言:javascript
复制
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
}
票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/33212623

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档