1

在打开的 DatePickerDialog 中,当屏幕方向改变时,它会重置选定的用户数据。

(DatePickerDialog 不会关闭,也不会维护选定的数据)

代码:

public class ActivityNeki extends FragmentActivity {
    DialogFragment newDF = null;
    private int datY, datM, datD;

    @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState);
        setContentView(R.layout.activity);
        if(savedInstanceState == null){ setTheData(); writeTheData(); }
    }

    @Override protected void onSaveInstanceState(Bundle outState) {
        outState.putInt("izY", datY); outState.putInt("izM", datM); outState.putInt("izD", datD);
        super.onSaveInstanceState(outState);
    }
    @Override protected void onRestoreInstanceState(Bundle savedInstanceState) { super.onRestoreInstanceState(savedInstanceState);
        datY = savedInstanceState.getInt("izY"); datM = savedInstanceState.getInt("izM"); datD = savedInstanceState.getInt("izD");
        writeTheData();
    }

    public void onClickOpenDPD(View view) {    // the method that is caled from XML onClick
        class MyDialogFragment extends DialogFragment {
            @Override public void onDestroyView() {
                if (getDialog() != null && getRetainInstance()) getDialog().setDismissMessage(null);
                super.onDestroyView();
            }
            @Override public void onCreate(Bundle state) { super.onCreate(state);
                setRetainInstance(true);
            }
            @Override public Dialog onCreateDialog(Bundle state) {
                DatePickerDialog dpd = new DatePickerDialog( getActivity(), new DatePickerDialog.OnDateSetListener() {
                    @Override public void onDateSet(DatePicker view, int leto, int mesec, int dan) {
                        datY = leto; datM = mesec; datD = dan;
                        writeTheData();
                } }, datY, datM, datD);
                return dpd;
            }
        }
        newDF = new MyDialogFragment();
        newDF.show( getSupportFragmentManager(), null );
    }
    public void setTheData(){
        Calendar c = Calendar.getInstance();
        datY = c.get(Calendar.YEAR); datM = c.get(Calendar.MONTH); datD = c.get(Calendar.DAY_OF_MONTH);
    }
    public void writeTheData(){  /* writes the data in a txtView */ }
}

建议我,如何解决这个问题?

4

3 回答 3

2

你的代码看起来很糟糕。

首先,您在方法主体中声明匿名片段类 - 但是,最好将此类与Activity类分开或使其成为静态内部类。

然后, yousetRetainInstance(true)告诉 Android 在 Activity 旋转时不要销毁片段。这适用于其他用例,您的情况并非如此。

我建议您阅读以下信息:

处理运行时更改

活动生命周期

如何使用 Fragments 通信 Activity

基本上,如果您按照 Android 规则进行游戏,大部分用户输入将在设备旋转时为您保存开箱即用(请参阅上面的链接)。

于 2015-02-05T10:29:59.663 回答
1

正确实施的 DialogFragment 将能够在您无需任何额外工作的情况下恢复状态。您的代码中的主要问题是非静态内部类。这是有缺陷的,原因有两个:

  1. 非静态内部类自然会保留对其外部类的引用(参见#2 中的链接)。这实际上是对您的活动、您的背景、观点的参考;基本上用户面临的一切。当设备旋转或用户切换到另一个应用程序时,您的应用程序很可能会关闭。但是,保留的片段将保持活动状态,这有时非常有用。但由于在您的情况下它是一个内部类,因此不允许垃圾收集丢弃上下文、活动等,因为周围仍然有一个活动的引用。因此,您正在泄漏内存。如果您在应用程序重新创建后访问此上下文(例如更新视图),您将收到运行时异常,告诉您您正在处理过时的上下文/视图。你可能想看看WeakReferences 如果你真的想走那条路。

    https://stackoverflow.com/a/10968689/1493269

  2. 非静态内部类危险的另一个原因是它们从不提供默认构造函数。但是 Android 框架要求每个 Fragment 都有一个空的构造函数,以便在后台实例化它。如果您的应用程序被拆除并重新启动(例如在旋转更改时),那么 Android 将通过强制默认构造函数重新实例化您的 Fragment,然后通过序列化包恢复状态。如果没有默认构造函数:运行时异常。你可以通过保留你的片段来解决这个问题,但我在 1) 中解释了为什么这也很糟糕。

    http://thecodersbreakfast.net/index.php?post/2011/09/26/Inner-classes-and-the-myth-of-the-default-constructor

故事的寓意:小心非静态内部类。

您的代码应该如何工作:

ActivityNeki.java

public class ActivityNeki extends FragmentActivity implements DatePickerDialog.OnDateSetListener
{
    @Override
    protected void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity);

        Fragment fragment = getSupportFragmentManager().findFragmentByTag( "my_dialog_tag" );

        if( fragment != null )
        {
            ( (MyDialogFragment) fragment ).listener = this;
        }
    }

    // Called from xml
    public void onClickOpenDPD(View view)
    {    
       MyDialogFragment.newInstance( x, x, x, this ).show( getSupportFragmentManager(), "my_dialog_tag" );
    }

    public void onDateSet(DatePicker view, int year, int monthOfYear, int dayOfMonth)
    {
        // Do your crazy callback stuff
    }
}

MyDialogFragment.java

class MyDialogFragment extends DialogFragment
{
    public static MyDialogFragment newInstance( int datY, int datM, int datD, DatePickerDialog.OnDateSetListener listener )
    {
        Bundle bundle = new Bundle( 3 );
        bundle.putInt( "y", datY );
        bundle.putInt( "m", datM );
        bundle.putInt( "d", datD );

        MyDialogFragment fragment = new MyDialogFragment();
        fragment.setArguments( bundle );
        fragment.listener = listener;
        return fragment;
    }

    public DatePickerDialog.OnDateSetListener listener = null;

    // Not entirely sure if this is still necessary    
    @Override
    public void onDestroyView()
    {
        if(getDialog() != null && getRetainInstance())
        {
            getDialog().setDismissMessage(null);
            super.onDestroyView();
        }
    }

    @Override
    public Dialog onCreateDialog(Bundle state)
    {
        return new DatePickerDialog(
            getActivity(),
            listener,
            getArguments().getInt( "y" ),
            getArguments().getInt( "m" ),
            getArguments().getInt( "d" )
        );
    }
}

请原谅小错误,因为我在没有测试的情况下就写下了这个。

于 2015-02-05T16:34:29.280 回答
0

当方向改变时,应用程序实际上被销毁并再次创建。所以你必须将状态保存在某个地方,例如在 instantState Bundle 中

-- 新编辑

我尝试了一种不同的方法,在 Button 中使用 OnClickListener 回调,并且状态保持不变。

@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Button button1 = (Button) findViewById(R.id.button1);

        button1.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                // the method that is called from XML onClick
                class MyDialogFragment extends DialogFragment {
                    @Override
                    public void onDestroyView() {
                        if (getDialog() != null && getRetainInstance())
                            getDialog().setDismissMessage(null);
                        super.onDestroyView();
                    }

                    @Override
                    public void onCreate(Bundle state) {
                        super.onCreate(state);
                        setRetainInstance(true);
                    }

                    @Override
                    public Dialog onCreateDialog(Bundle state) {
                        DatePickerDialog dpd = new DatePickerDialog(getActivity(), new DatePickerDialog.OnDateSetListener() {
                            @Override
                            public void onDateSet(DatePicker view, int leto, int mesec, int dan) {
                                datY = leto;
                                datM = mesec;
                                datD = dan;
                                writeTheData();
                            }
                        }, datY, datM, datD);
                        return dpd;
                    }
                }
                newDF = new MyDialogFragment();
                newDF.show(getFragmentManager(), null);
            }
        });

    }
于 2015-02-04T16:54:21.733 回答