Действительно ли фрагментам нужен пустой конструктор?

У меня есть Fragment с конструктором, который принимает несколько аргументов. Мое приложение отлично работало во время разработки, но в производстве мои пользователи иногда видят этот сбой:

android.support.v4.app.Fragment$InstantiationException: Unable to instantiate fragment 
make sure class name exists, is public, and has an empty constructor that is public

Я мог бы создать пустой конструктор, как это предлагает сообщение об ошибке, но это не имеет смысла для меня, с тех пор мне придется вызвать отдельный метод, чтобы завершить настройку Fragment.

Мне любопытно, почему этот крах случается только иногда. Может быть, я неправильно использую ViewPager? Я сам создаю экземпляр всего Fragment и сохраняю его в списке внутри Activity. Я не использую транзакции FragmentManager, так как примеры ViewPager я видел, не нуждались в этом, и все, казалось, работало во время разработки.

Ответ 1

Да, они делают.

В любом случае, вы не должны переопределять конструктор. Вы должны иметь статический метод newInstance() и передавать любые параметры через аргументы (bundle)

Например:

public static final MyFragment newInstance(int title, String message) {
    MyFragment f = new MyFragment();
    Bundle bdl = new Bundle(2);
    bdl.putInt(EXTRA_TITLE, title);
    bdl.putString(EXTRA_MESSAGE, message);
    f.setArguments(bdl);
    return f;
}

И, конечно же, схватил арги так:

@Override
public void onCreate(Bundle savedInstanceState) {
    title = getArguments().getInt(EXTRA_TITLE);
    message = getArguments().getString(EXTRA_MESSAGE);

    //...
    //etc
    //...
}

Затем вы должны создать экземпляр из вашего менеджера фрагментов следующим образом:

@Override
public void onCreate(Bundle savedInstanceState) {
    if (savedInstanceState == null){
        getSupportFragmentManager()
            .beginTransaction()
            .replace(R.id.content, MyFragment.newInstance(
                R.string.alert_title,
                "Oh no, an error occurred!")
            )
            .commit();
    }
}

Таким образом, если отсоединено и повторно присоединено, состояние объекта может быть сохранено через аргументы. Очень похоже на связки, прикрепленные к Intents.

Причина - Дополнительное чтение

Я думал, что объясню, почему людям интересно, почему.

Если вы проверите: https://android.googlesource.com/platform/frameworks/base/+/master/core/java/android/app/Fragment.java

Вы увидите, что метод instantiate(..) в классе Fragment вызывает метод newInstance:

public static Fragment instantiate(Context context, String fname, @Nullable Bundle args) {
    try {
        Class<?> clazz = sClassMap.get(fname);
        if (clazz == null) {
            // Class not found in the cache, see if it real, and try to add it
            clazz = context.getClassLoader().loadClass(fname);
            if (!Fragment.class.isAssignableFrom(clazz)) {
                throw new InstantiationException("Trying to instantiate a class " + fname
                        + " that is not a Fragment", new ClassCastException());
            }
            sClassMap.put(fname, clazz);
        }
        Fragment f = (Fragment) clazz.getConstructor().newInstance();
        if (args != null) {
            args.setClassLoader(f.getClass().getClassLoader());
            f.setArguments(args);
        }
        return f;
    } catch (ClassNotFoundException e) {
        throw new InstantiationException("Unable to instantiate fragment " + fname
                + ": make sure class name exists, is public, and has an"
                + " empty constructor that is public", e);
    } catch (java.lang.InstantiationException e) {
        throw new InstantiationException("Unable to instantiate fragment " + fname
                + ": make sure class name exists, is public, and has an"
                + " empty constructor that is public", e);
    } catch (IllegalAccessException e) {
        throw new InstantiationException("Unable to instantiate fragment " + fname
                + ": make sure class name exists, is public, and has an"
                + " empty constructor that is public", e);
    } catch (NoSuchMethodException e) {
        throw new InstantiationException("Unable to instantiate fragment " + fname
                + ": could not find Fragment constructor", e);
    } catch (InvocationTargetException e) {
        throw new InstantiationException("Unable to instantiate fragment " + fname
                + ": calling Fragment constructor caused an exception", e);
    }
}

http://docs.oracle.com/javase/6/docs/api/java/lang/Class.html#newInstance() Объясняет, почему после создания экземпляра он проверяет, что метод доступа является public и что загрузчик классов разрешает доступ к нему,

Это довольно неприятный метод, но он позволяет FragmentManger уничтожать и воссоздавать Fragments с состояниями. (Подсистема Android делает то же самое с Activities).

Пример класса

Меня часто спрашивают о вызове newInstance. Не путайте это с методом класса. Весь этот пример класса должен показать использование.

/**
 * Created by chris on 21/11/2013
 */
public class StationInfoAccessibilityFragment extends BaseFragment implements JourneyProviderListener {

    public static final StationInfoAccessibilityFragment newInstance(String crsCode) {
        StationInfoAccessibilityFragment fragment = new StationInfoAccessibilityFragment();

        final Bundle args = new Bundle(1);
        args.putString(EXTRA_CRS_CODE, crsCode);
        fragment.setArguments(args);

        return fragment;
    }

    // Views
    LinearLayout mLinearLayout;

    /**
     * Layout Inflater
     */
    private LayoutInflater mInflater;
    /**
     * Station Crs Code
     */
    private String mCrsCode;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mCrsCode = getArguments().getString(EXTRA_CRS_CODE);
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        mInflater = inflater;
        return inflater.inflate(R.layout.fragment_station_accessibility, container, false);
    }

    @Override
    public void onViewCreated(View view, Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        mLinearLayout = (LinearLayout)view.findViewBy(R.id.station_info_accessibility_linear);
        //Do stuff
    }

    @Override
    public void onResume() {
        super.onResume();
        getActivity().getSupportActionBar().setTitle(R.string.station_info_access_mobility_title);
    }

    // Other methods etc...
}

Ответ 2

Как отмечено CommonsWare в этом вопросе fooobar.com/questions/30679/..., эта ошибка также может возникнуть, если вы создаете анонимный подкласс фрагмента, поскольку анонимные классы не могут иметь конструкторы.

Не создавать анонимные подклассы фрагмента: -)

Ответ 3

Да, как вы можете видеть, пакет поддержки также создает фрагменты (когда они уничтожаются и открываются заново). Ваши подклассы Fragment нуждаются в открытом пустом конструкторе, поскольку это то, что вызывается фреймворком.

Ответ 4

Вот мое простое решение:

1 - Определите фрагмент

public class MyFragment extends Fragment {

    private String parameter;

    public MyFragment() {
    }

    public void setParameter(String parameter) {
        this.parameter = parameter;
    } 
}

2 - Создайте новый фрагмент и запишите параметр

    myfragment = new MyFragment();
    myfragment.setParameter("here the value of my parameter");

3 - Наслаждайтесь!

Очевидно, вы можете изменить тип и количество параметров. Быстро и легко.