片段 - 实例化异常:没有空构造函数 -> 谷歌地图v2?

17
当我通过应用程序切换按钮重新打开已关闭的应用程序时,会出现以下错误消息:
Caused by: java.lang.InstantiationException: can't instantiate class com.*.FragmentContact$1; no empty constructor

我找到了一些有关内部类和使它们静态等的提示。但是,FragmentContact是一个公共类,在一个*.java文件中,并且有一个公共空构造函数。 我在此项目中使用Google Maps Api v2,并从互联网上的某个地方进行了一些技巧来设置我的MapView。看这里:

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View v = inflater.inflate(R.layout.fragment_contact, null);

    try {
        MapsInitializer.initialize(this.getActivity().getApplicationContext());
    } catch (GooglePlayServicesNotAvailableException e) {
        e.printStackTrace();
    }

    fragment = new SupportMapFragment() {
        @Override
        public void onActivityCreated(Bundle savedInstanceState) {
            super.onActivityCreated(savedInstanceState);
            mMap = fragment.getMap();
            if (mMap != null) {
                setupMap();
            }
        }
    };

    getFragmentManager().beginTransaction().replace(R.id.fragment_orte_map_parent, fragment).commit();

    return v;
}

当我将这个MapView的部分删除后,一切都正常。也许有人可以解释一下我做错了什么。
完整的堆栈跟踪:
FATAL EXCEPTION: main
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.*/com.*.MainActivity}: android.support.v4.app.Fragment$InstantiationException: Unable to instantiate fragment com.*.FragmentContact$1: make sure class name exists, is public, and has an empty constructor that is public
   at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2307)
   at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2357)
   at android.app.ActivityThread.access$600(ActivityThread.java:153)
   at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1247)
   at android.os.Handler.dispatchMessage(Handler.java:99)
   at android.os.Looper.loop(Looper.java:137)
   at android.app.ActivityThread.main(ActivityThread.java:5226)
   at java.lang.reflect.Method.invokeNative(Native Method)
   at java.lang.reflect.Method.invoke(Method.java:511)
   at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:795)
   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:562)
   at dalvik.system.NativeStart.main(Native Method)
Caused by: android.support.v4.app.Fragment$InstantiationException: Unable to instantiate fragment com.*.FragmentContact$1: make sure class name exists, is public, and has an empty constructor that is public
   at android.support.v4.app.Fragment.instantiate(Fragment.java:405)
   at android.support.v4.app.FragmentState.instantiate(Fragment.java:97)
   at android.support.v4.app.FragmentManagerImpl.restoreAllState(FragmentManager.java:1767)
   at android.support.v4.app.FragmentActivity.onCreate(FragmentActivity.java:208)
   at com.*.MainActivity.onCreate(MainActivity.java:20)
   at android.app.Activity.performCreate(Activity.java:5104)
   at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1080)
   at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2261)
   ... 11 more
Caused by: java.lang.InstantiationException: can't instantiate class com.*.FragmentContact$1; no empty constructor
   at java.lang.Class.newInstanceImpl(Native Method)
   at java.lang.Class.newInstance(Class.java:1319)
   at android.support.v4.app.Fragment.instantiate(Fragment.java:394)
   ... 18 more
4个回答

32
但是,这个 FragmentContact 是一个公共类(*.java 文件中)并且具有一个公共的空构造函数。
错误并没有抱怨 FragmentContact。它抱怨了 FragmentContact 的第一个内部类(FragmentContact$1)。您不能将 Fragment 实现为任何东西的内部类,因为它无法从外部类实例化。静态内部类是可以的。

1
抱歉,是我的错。我没有识别匿名内部类,因为我一直在想非匿名内部类。你说得对。我重写了 public void onActivityCreated(Bundle savedInstanceState) 方法,所以我使用了一个匿名内部类。真是糟糕... - Osiriz
5
@Osiriz: $1 的意思是它是一个匿名内部类,否则它将会是 $WhateverYouCalledTheInnerClass。话虽如此,在最初阅读您的代码时我错过了这一点,认为它在其他文件中。 - CommonsWare
3
我必须承认,直到现在我甚至不知道堆栈跟踪中的“$1”是什么意思。非常感谢! - Osiriz
你好,我遇到了相同的问题...我并不真正明白解释,也不知道如何修复它...避免这个问题最简单的方法是什么? - Waza_Be
2
@Waza_Be:不要创建常规(非静态)内部类作为片段。它们需要是静态内部类或常规(独立的)Java类。 - CommonsWare
显示剩余6条评论

0

我已经遇到这个问题很长时间了,终于在这里的评论中找到了解决方法。关于$1表示匿名内部类的信息很有用。

将地图初始化移动到一个新的公共类中,例如MyMapFragment.java。这将允许片段在没有实例的情况下被实例化。

对于那些仍然不确定如何修复它的人,请按照以下方式编写代码(使用原始问题中的示例代码):

//Existing code to instantiate map
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

    View v = inflater.inflate(R.layout.fragment_contact, null);

    try {
        MapsInitializer.initialize(this.getActivity().getApplicationContext());
    } catch (GooglePlayServicesNotAvailableException e) {
        e.printStackTrace();
    }

    fragment = new MyMapFragment();

    //added callback for catching when the map has finished loading
    mapFragment.setLoadedCallback(new MapLoadedCallback() {
        @Override
        public void onLoaded(GoogleMap map) {
            mMap = map;
        }
    });

    getFragmentManager().beginTransaction().replace(R.id.fragment_orte_map_parent, fragment).commit();

    return v;
}
....

创建新的 MyMapFragment.java 文件,使其不再是内部类。这将允许片段在没有外部类的情况下被创建。
public class MyMapFragment extends SupportMapFragment 
{
    GoogleMap mMap;
    public MyMapFragment() {
    }

    @Override
    public void onActivityCreated(Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
         mMap = fragment.getMap();
         if (mMap != null) {
             mCallback.onLoaded(mMap);
        }
    }
}    

创建 MapLoadedCallback.java 以便您可以处理地图加载完成时的操作,并检索已加载地图对象的实例。
public interface MapLoadedCallback {
    void onLoaded(GoogleMap map);
}

这并没有帮助,因为这不是静态内部类。唯一的区别是这个不是匿名的。 - Selvin
@Selvin:我不确定你的意思。问题已经解决,因为该类不再是匿名的。如果它是静态的,问题本来就不存在了,因为静态内部类可以被正确解析。 - Matthew Cawley
内部非静态类仍然没有空构造函数(在外部类范围之外)...您不能调用new Outer.Inner()(是的,在Outer内部可以调用new Inner(),但在Fragment.instantiate中无法这样做) - 换句话说:Java的基础知识...内部非静态类实例不能在没有外部类实例的情况下初始化。 - Selvin
是的,我现在明白你的意思了。在回答后,我对实现方式进行了更改。我会更新代码以匹配我的更改。感谢你指出这一点。请给我几分钟时间来更新我的回答。 - Matthew Cawley
我已经更新了我的答案,希望现在更好了。再次感谢@Selvin指出这一点。 - Matthew Cawley

0
  1. 理想情况下,我们不应该在片段构造函数中传递任何内容,片段构造函数应该是空的或默认的。

  2. 现在第二个问题是,如果我们想要传递接口变量或参数-

    • a. 我们应该使用Bundle来传递数据。
    • b. 对于接口,我们可以将其放入bundle中并使该接口实现parceble
    • c. 如果可能,我们可以在activity中实现该接口,并在fragment中初始化listener,在OnAttach中我们有context[(context) Listener]。

这样,在配置更改(例如字体更改、Activity重新创建)期间,listener不会未初始化,我们可以避免空指针异常。


-3
  private Entity mEntity;
  public YourFragment() {}
  public static YourFragment getInstance(Entity mEntity) {
    YourFragment fragment = new YourFragment();
    fragment.mEntity = mEntity;
    return fragment;
  }

当您使用此片段时,只需调用静态方法getInstance


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接