在应用程序的启动过程中,我有一些需要异步完成的事情。我创建了一个派生自AsyncTask的抽象类,它实现了这些加载任务的共同点:
protected abstract class LoadTask extends AsyncTask<Object, Void, Object>{
MainActivity activity;
void detach(){
activity = null;
}
void attach(MainActivity a){
activity = a;
}
public LoadTask(MainActivity a){
super();
attach(a);
}
public abstract boolean shouldRun();
@Override
protected void onPostExecute(Object result){
activity.onFinishLoad(this.getClass(), result);
}
}
以下是具体子类的一个特定实现:
public class ItemInfoLoadTask extends LoadTask{
public ItemInfoLoadTask(MainActivity a){
super(a);
}
public boolean shouldRun(){
return should_reload("items", item_info_path);
}
@Override
protected Object doInBackground(Object ... params) {
setProgressMessage("Reading item info...");
JSONObject result = loadItemInfo();
return result;
}
private JSONObject loadItemInfo(){
return Util.parseJSONObject(item_info_path);
}
}
以下是从我的活动的onCreate()方法中调用这些代码的代码:
Log.v(TAG, c.getName());
Class[] types = {MainActivity.class};
Constructor construct = c.getConstructor(types);
task = (LoadTask)(construct.newInstance(this));
这是日志:
12-17 10:26:25.750: V/MainActivity(25188): com.mycompany.mypackage.MainActivity$ItemInfoLoadTask
12-17 10:26:25.760: E/MainActivity(25188): this shouldn't happen
12-17 10:26:25.760: E/MainActivity(25188): java.lang.NoSuchMethodException: <init> [class com.mycompany.mypackage.MainActivity]
12-17 10:26:25.760: E/MainActivity(25188): at java.lang.ClassMembers.getConstructorOrMethod(ClassMembers.java:235)
12-17 10:26:25.760: E/MainActivity(25188): at java.lang.Class.getConstructor(Class.java:459)
12-17 10:26:25.760: E/MainActivity(25188): at com.mycompany.mypackage.MainActivity.onCreate(MainActivity.java:80)
12-17 10:26:25.760: E/MainActivity(25188): at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1048)
12-17 10:26:25.760: E/MainActivity(25188): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1715)
12-17 10:26:25.760: E/MainActivity(25188): at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1767)
12-17 10:26:25.760: E/MainActivity(25188): at android.app.ActivityThread.access$1500(ActivityThread.java:122)
12-17 10:26:25.760: E/MainActivity(25188): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1005)
12-17 10:26:25.760: E/MainActivity(25188): at android.os.Handler.dispatchMessage(Handler.java:99)
12-17 10:26:25.760: E/MainActivity(25188): at android.os.Looper.loop(Looper.java:132)
12-17 10:26:25.760: E/MainActivity(25188): at android.app.ActivityThread.main(ActivityThread.java:4028)
12-17 10:26:25.760: E/MainActivity(25188): at java.lang.reflect.Method.invokeNative(Native Method)
12-17 10:26:25.760: E/MainActivity(25188): at java.lang.reflect.Method.invoke(Method.java:491)
12-17 10:26:25.760: E/MainActivity(25188): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:844)
12-17 10:26:25.760: E/MainActivity(25188): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:602)
12-17 10:26:25.760: E/MainActivity(25188): at dalvik.system.NativeStart.main(Native Method)
我无法弄清楚我做错了什么。如果它重要(尽管我不明白为什么它应该),LoadTask和ItemInfoLoadTask是MainActivity中的内部类
答案 0 :(得分:3)
非静态内部类包含对封闭类实例的隐式引用。这些内部类的所有构造函数都引用封闭类的实例作为参数。
一个简单的例子:
import java.lang.reflect.*;
class Foo {
class Bar {
public Bar(int a) {
}
}
public static void main(String[] args) {
for(Constructor c: Bar.class.getDeclaredConstructors()) {
System.out.println(c);
}
}
}
打印public Foo$Bar(Foo,int)
- 请注意第一个Foo
参数。
javap工具可以显示有关结构的信息 - Foo$Bar.class
声明的字段,方法和构造函数:
Compiled from "Foo.java"
class Foo$Bar extends java.lang.Object{
final Foo this$0;
public Foo$Bar(Foo, int);
}
在这里,您可以看到对父级的隐式引用。 JVM不了解内部类,因此编译器必须引入合成方法和字段来实现所需的行为。在你需要反射之类的东西之前,它通常会被忽视。
在您的特定情况下,您可以将内部类设置为静态,这样就不会有隐式引用或删除对MainActivity
的显式引用。