画面回転後のフラグメントの再作成を回避するための実際のアプローチを見つけています
if (container == null) { return null; }
フラグメントが再作成されることを実際に回避しているわけではありません。(下図)
公式フラグメント開発者ガイドはどこにありますか?
私たちが関わっている公式ガイドはhttp://developer.android.com/guide/components/fragments.htmlにあります。部分的なサンプルコードはガイドの下部にあります。私の知る限り、完全なサンプルコードは、Android 3.0(API 11)の「SamplesforSDK」で入手できます。また、API 10-で実行するためにサンプルコードに最小限の変更を加え、この質問の下部に含まれているいくつかのデバッグメッセージを追加しました。
どこにありR.id.a_item
ますか?
開発者ガイドの例で、次のコードスタブを見つけることができます。
if (index == 0) {
ft.replace(R.id.details, details);
} else {
ft.replace(R.id.a_item, details);
}
私はインターネットでいくつかの検索をしました、そして他のいくつかがまたどこにあるかについて関係しているのを見つけましたR.id.a_item
。API 11でサンプルを調べた後、それは意味のないタイプミスであると確信しています。サンプルにはそのような線はまったくありません。
画面が回転した後のフラグメントの再作成を回避するための実際のアプローチは?
ウェブ上には多くの既存の議論があります。しかし、まだ「本当の」解決策はないようです。
DetailsFragment
クラスのライフサイクルを追跡するために、以下のコードに多くのデバッグメッセージを追加しました。(1)プログラムをポートレートモードで開始し、(2)デバイスをランドスケープモードに切り替え、(3)デバイスをポートレートに戻し、(4)再びランドスケープに、(5)ポートレートに戻し、最後に(6)終了します。次のデバッグメッセージが表示されます。
(1)ポートレートモードで開始します
TitlesFragment.onCreate()Bundle = null
のみTitlesFragment
作成されます。DetailsFragment
まだ表示されていません。
(2)ランドスケープモードに切り替える
TitlesFragment.onCreate()Bundle = Bundle [{shownChoice = -1、android:view_state = android.util.SparseArray @ 4051d3a8、curChoice = 0}]
DetailsFragment.onAttach()Activity = com.example.android.apis.app.FragmentLayout @ 4051d640
DetailsFragment.onCreate()Bundle = null
DetailsFragment.onCreateView()Activity =
android.widget.FrameLayout@4050df68 DetailsFragment.onActivityCreated()Bundle = null
DetailsFragment.onStart()
DetailsFragment.onResume()
まず、TitlesFragment
が再作成されます(savedInstanceStateバンドルを使用)。次に、DetailsFragment
動的に作成されます(TitlesFragment.onActivityCreated()
、呼び出しshowDetails()
、を使用してFragmentTransaction
)。
(3)ポートレートモードに戻る
DetailsFragment.onPause()
DetailsFragment.onStop()
DetailsFragment.onDestroyView()
DetailsFragment.onDestroy()
DetailsFragment.onDetach()
DetailsFragment.onAttach()Activity
= com.example.android.apis.app.FragmentLayout@40527f70 DetailsFragment.onCreate() Bundle = null
TitlesFragment.onCreate()Bundle = Bundle [{shownChoice = 0、android:view_state = android.util.SparseArray @ 405144b0、curChoice = 0}]
DetailsFragment.onCreateView()Activity = null
DetailsFragment.onActivityCreated()Bundle = null
DetailsFragment.onStart()
DetailsFragment.onResume()
これが、実際の再作成回避アプローチについて私たちが懸念している最初の場所です。
これは、以前はランドスケープモードでDetailsFragment
アタッチされていたためです。layout-land/fragment_layout.xml
<FrameLayout>
ViewGroup
そしてそれはID(R.id.details
)を持っています。画面が回転すると、ViewGroup
のインスタンスであるDetailsFragment
、がFragmentLayoutのonSaveInstanceState()のActivityFragmentLayoutのバンドルに保存されます。ポートレートモードに入った後、DetailsFragment
が再作成されます。ただし、ポートレートモードでは必要ありません。
サンプルでは(そして他の多くの人が示唆しているように)、DetailsFragment
クラスはポートレートモードで表示されないようにするためにを使用if (container == null) { return null; }
しています。ただし、上記のデバッグメッセージに示されているように、は、すべてのライフサイクルメソッド呼び出しを持って、孤立した状態でバックグラウンドでまだ生きています。onCreateView()
DetailsFragment
DetailsFragment
(4)再びランドスケープモードに
DetailsFragment.onPause()
DetailsFragment.onStop()
DetailsFragment.onDestroyView()
DetailsFragment.onDestroy()
DetailsFragment.onDetach()
DetailsFragment.onAttach()Activity
= com.example.android.apis.app.FragmentLayout@4052c7d8 DetailsFragment.onCreate() Bundle = null
TitlesFragment.onCreate()Bundle = Bundle [{shownChoice = 0、android:view_state = android.util.SparseArray @ 40521b80、curChoice = 0}]
DetailsFragment.onCreateView()Activity =
android.widget.FrameLayout @40525270DetailsFragment。 onActivityCreated()Bundle = null
DetailsFragment.onStart()
DetailsFragment.onResume()
最初の5行で、DetailsFragment
ライフサイクル状態が完了し、破棄されて切り離されていることに注意してください。
if (container == null) { return null; }
これは、メソッドがインスタンスを取り除くための実際のアプローチではないことをさらに証明します。(ガベージコレクターがこのぶら下がっている子を破壊すると思いましたが、そうではありませんでした。Androidがぶら下がっているフラグメントを許可しているためです。参照:UIなしでフラグメントを追加します。)DetailsFragment
私が理解している限りでは、6行目から、(2)で行ったように、DetailsFragment
によって作成された新しいインスタンスである必要があります。TitlesFragment
しかし、なぜDetailsFragment
'sonAttach()
とonCreate()
メソッドがTitlesFragment
'sの前に呼び出されるのか説明できませんonCreate()
。
しかし、のバンドルnull
は、それが新しいインスタンスであることを証明します。DetailsFragment
onCreate()
私の理解では、以前のぶら下がっているDetailsFragment
インスタンスはIDを持っていないため、今回は再作成されていません。そのため、ビュー階層とともにsavedInstanceStateバンドルに自動保存されませんでした。
(5)再びポートレートモードに戻る
DetailsFragment.onPause()
DetailsFragment.onStop()
DetailsFragment.onDestroyView()
DetailsFragment.onDestroy()
DetailsFragment.onDetach()
DetailsFragment.onAttach()Activity
= com.example.android.apis.app.FragmentLayout@4052d7d8 DetailsFragment.onCreate() Bundle = null
TitlesFragment.onCreate()Bundle = Bundle [{shownChoice = 0、android:view_state = android.util.SparseArray @ 40534e30、curChoice = 0}]
DetailsFragment.onCreateView()Activity = null
DetailsFragment.onActivityCreated()Bundle = null
DetailsFragment.onStart()
DetailsFragment.onResume()
ここで、すべてのライフサイクルコールバックは、とが異なることを除いて、(3)で初めてポートレートに戻ったときと同じであることに注意してください。これは合理的です。FragmentLayoutアクティビティとインスタンス状態バンドルの両方が再作成されます。Activity ID (40527f70 vs 4052d7d8)
view_state Bundle (405144b0 vs 40534e30)
(6)終了(戻るボタンによる)
I / System.out(29845):DetailsFragment.onPause()I / System.out(29845):DetailsFragment.onStop()I / System.out(29845):DetailsFragment.onDestroyView()I / System.out(29845) :DetailsFragment.onDestroy()I / System.out(29845):DetailsFragment.onDetach()
のを削除できれば完璧DetailsFragment
です。ただし、の前に'sメソッドを呼び出す必要があります。ただし、で画面が回転しているかどうかを判断する方法はありません。FragmentLayout
onDestroy()
FragmentTransaction
remove()
onSaveInstanceState()
onSaveInstanceState()
DetailsFragment
とにかくinを削除することもできませFragmentLayout
んonSaveInstanceState()
。まず、DetailsFragment
ダイアログボックスによって部分的に隠されている場合は、バックグラウンドで非表示になります。また、ダイアログボックスで隠されたり、アクティビティを切り替えたりした場合は、再度呼び出されることonCreate(Bundle)
もありません。onRestoreInstanceState(Bundle)
したがって、フラグメントを復元する(およびバンドルからデータを取得する)場所がありません。
ソースコードとファイル
FragmentLayout.java
package com.example.android.apis.app;
import android.app.Activity;
import android.content.Intent;
import android.content.res.Configuration;
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentActivity;
import android.support.v4.app.FragmentTransaction;
import android.support.v4.app.ListFragment;
import android.util.TypedValue;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.ScrollView;
import android.widget.TextView;
public class FragmentLayout extends FragmentActivity {
private final static class Shakespeare {
public static final String[] TITLES = { "Love", "Hate", "One", "Day" };
public static final String[] DIALOGUE = {
"Love Love Love Love Love",
"Hate Hate Hate Hate Hate",
"One One One One One",
"Day Day Day Day Day" };
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.fragment_layout);
}
public static class DetailsActivity extends FragmentActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (getResources().getConfiguration().orientation
== Configuration.ORIENTATION_LANDSCAPE) {
// If the screen is now in landscape mode, we can show the
// dialog in-line with the list so we don't need this activity.
finish();
return;
}
if (savedInstanceState == null) {
// During initial setup, plug in the details fragment.
DetailsFragment details = new DetailsFragment();
details.setArguments(getIntent().getExtras());
getSupportFragmentManager().beginTransaction().add(android.R.id.content, details).commit();
}
}
}
public static class TitlesFragment extends ListFragment {
boolean mDualPane;
int mCurCheckPosition = 0;
int mShownCheckPosition = -1;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
System.out.println(getClass().getSimpleName() + ".onCreate() Bundle=" +
(savedInstanceState == null ? null : savedInstanceState));
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
// Populate list with our static array of titles.
setListAdapter(new ArrayAdapter<String>(getActivity(),
android.R.layout.simple_list_item_1, Shakespeare.TITLES));
// API 11:android.R.layout.simple_list_item_activated_1
// Check to see if we have a frame in which to embed the details
// fragment directly in the containing UI.
View detailsFrame = getActivity().findViewById(R.id.details);
mDualPane = detailsFrame != null && detailsFrame.getVisibility() == View.VISIBLE;
if (savedInstanceState != null) {
// Restore last state for checked position.
mCurCheckPosition = savedInstanceState.getInt("curChoice", 0);
mShownCheckPosition = savedInstanceState.getInt("shownChoice", -1);
}
if (mDualPane) {
// In dual-pane mode, the list view highlights the selected item.
getListView().setChoiceMode(ListView.CHOICE_MODE_SINGLE);
// Make sure our UI is in the correct state.
showDetails(mCurCheckPosition);
}
}
@Override
public void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
outState.putInt("curChoice", mCurCheckPosition);
outState.putInt("shownChoice", mShownCheckPosition);
}
@Override
public void onListItemClick(ListView l, View v, int position, long id) {
showDetails(position);
}
/**
* Helper function to show the details of a selected item, either by
* displaying a fragment in-place in the current UI, or starting a
* whole new activity in which it is displayed.
*/
void showDetails(int index) {
mCurCheckPosition = index;
if (mDualPane) {
// We can display everything in-place with fragments, so update
// the list to highlight the selected item and show the data.
getListView().setItemChecked(index, true);
if (mShownCheckPosition != mCurCheckPosition) {
// If we are not currently showing a fragment for the new
// position, we need to create and install a new one.
DetailsFragment df = DetailsFragment.newInstance(index);
// Execute a transaction, replacing any existing fragment
// with this one inside the frame.
FragmentTransaction ft = getFragmentManager().beginTransaction();
ft.replace(R.id.details, df);
ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE);
ft.commit();
mShownCheckPosition = index;
}
} else {
// Otherwise we need to launch a new activity to display
// the dialog fragment with selected text.
Intent intent = new Intent();
intent.setClass(getActivity(), DetailsActivity.class);
intent.putExtra("index", index);
startActivity(intent);
}
}
}
public static class DetailsFragment extends Fragment {
/**
* Create a new instance of DetailsFragment, initialized to
* show the text at 'index'.
*/
public static DetailsFragment newInstance(int index) {
DetailsFragment f = new DetailsFragment();
// Supply index input as an argument.
Bundle args = new Bundle();
args.putInt("index", index);
f.setArguments(args);
return f;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
System.out.println(getClass().getSimpleName() + ".onCreate() Bundle=" +
(savedInstanceState == null ? null : savedInstanceState));
}
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
System.out.println(getClass().getSimpleName() + ".onAttach() Activity=" +
(activity == null ? null : activity));
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
System.out.println(getClass().getSimpleName() + ".onActivityCreated() Bundle=" +
(savedInstanceState == null ? null : savedInstanceState));
}
@Override
public void onStart() { super.onStart(); System.out.println(getClass().getSimpleName() + ".onStart()"); }
@Override
public void onResume() { super.onResume(); System.out.println(getClass().getSimpleName() + ".onResume()"); }
@Override
public void onPause() { super.onPause(); System.out.println(getClass().getSimpleName() + ".onPause()"); }
@Override
public void onStop() { super.onStop(); System.out.println(getClass().getSimpleName() + ".onStop()"); }
@Override
public void onDestroyView() { super.onDestroyView(); System.out.println(getClass().getSimpleName() + ".onDestroyView()"); }
@Override
public void onDestroy() { super.onDestroy(); System.out.println(getClass().getSimpleName() + ".onDestroy()"); }
@Override
public void onDetach() { super.onDetach(); System.out.println(getClass().getSimpleName() + ".onDetach()"); }
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
System.out.println(getClass().getSimpleName() + ".onCreateView() Activity=" +
(container == null ? null : container));
if (container == null) {
// We have different layouts, and in one of them this
// fragment's containing frame doesn't exist. The fragment
// may still be created from its saved state, but there is
// no reason to try to create its view hierarchy because it
// won't be displayed. Note this is not needed -- we could
// just run the code below, where we would create and return
// the view hierarchy; it would just never be used.
return null;
}
ScrollView scroller = new ScrollView(getActivity());
TextView text = new TextView(getActivity());
int padding = (int)TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP,
4, getActivity().getResources().getDisplayMetrics());
text.setPadding(padding, padding, padding, padding);
scroller.addView(text);
text.setText(Shakespeare.DIALOGUE[getArguments().getInt("index", 0)]);
return scroller;
}
}
}
layout /fragment_layout.xml
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent" android:layout_height="match_parent">
<fragment class="com.example.android.apis.app.FragmentLayout$TitlesFragment"
android:id="@+id/titles"
android:layout_width="match_parent" android:layout_height="match_parent" />
</FrameLayout>
レイアウト-土地/fragment_layout.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="horizontal" android:baselineAligned="false"
android:layout_width="match_parent" android:layout_height="match_parent">
<fragment class="com.example.android.apis.app.FragmentLayout$TitlesFragment"
android:id="@+id/titles" android:layout_weight="1"
android:layout_width="0px" android:layout_height="match_parent" />
<FrameLayout android:id="@+id/details" android:layout_weight="1"
android:layout_width="0px" android:layout_height="match_parent" />
<!-- API 11:android:background="?android:attr/detailsElementBackground" -->
</LinearLayout>
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.android.apis.app"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="4"
android:targetSdkVersion="17" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.example.android.apis.app.FragmentLayout"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name="com.example.android.apis.app.FragmentLayout$DetailsActivity"
android:label="@string/app_name" >
</activity>
</application>
</manifest>