gpt4 book ai didi

android - 使用 DexClassLoader 与动态加载的库交互

转载 作者:行者123 更新时间:2023-11-30 01:49:58 25 4
gpt4 key购买 nike

我在运行时使用 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。我知道为什么我们会得到这个异常,它背后的原因是在打印 toast 时使用它的上下文中的空指针异常。所以它不是

的副本

What could cause java.lang.reflect.InvocationTargetException?

背后的原因是

  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 卡加载我的图书馆..我想随时更新图书馆在用户不知情且没有任何版本更新的情况下。库包括很少的界面、 fragment 和 Activity 。所以现在我可以从 SD 卡加载我的库并可以调用基本函数。现在的主要挑战是从库中实现接口(interface)并调用在其中使用上下文的函数。

涉及此类操作的任何示例或提示都会很有帮助。

最佳答案

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

  • 如果 MyClass 必须是一个 Activity

必须使用 startActivity() 或任何变体正确启动 Activity。它还必须在您的 list 中声明。因此,仅当您的所有 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

在这种情况下无需创建另一个 ActivitydoSomething() 只需要接收适当的 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.");
}
}

并更改启动代码以将 this 传递给 doSomething(),假设它是从 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
}

关于android - 使用 DexClassLoader 与动态加载的库交互,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33212623/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com