Android在3.0中引入了Fragments的概念,其目的是用在大屏幕设备上–例如平板电脑上,支持更加动态和灵活的UI设计。平板电脑的屏幕要比手机的大得多,有更多的空间来放更多的UI组件,并且这些组件之间会产生更多的交互。Fragment允许这样的一种设计,而不需要你亲自来管理 Viewhierarchy的复杂变化。 通过将Activity的布局分散到Fragment中, 你可以在运行时修改Activity的外观,并在由Activity管理的back stack中保存那些变化。
Android开发过程中,我们可以把Fragment想成Activity中的模块,这个模块有自己的布局,有自己的生命周期,单独处理自己的输入,在Activity运行的时候可以加载或者移除Fragment模块。从一定程度上来说,与Web开发中的iframe和DIV布局思想有点类似。例像网易新闻、易车等应用程序将一个fragment放在左边显示文章列表,在右边用另一个Fragment来显示一篇文章——两个Fragment在同一个Activity中并排着,并且每个Fragment都有其自己的生命周期回调方法序列用以处理各自的用户输入事件。因此,用户可以在同一个Activity中选择和阅读文章,而不是在一个Activity中选择,在另一个Activity中阅读。这样设计的好处就是当程序运行在平板尺寸屏幕设备上时,可以在Activity A中嵌入两个Fragment。但是,当运行在手机尺寸屏幕上,就没有足够的空间容纳两个Fragment了,因此Activity A只能引用包含文章列表的Fragment,在当用户选择一篇文章时,可以启动Activity B,它包含了用来阅读文章的第二个fragment。这样,应用程序通过以不同组合的复用fragments支持了平板电脑和手机,再比如官方的这个例子:
因为Fragment必须嵌入在Acitivity中使用,所以Fragment的生命周期和它所在的Activity是密切相关的。如果Activity是暂停状态,其中所有的Fragment都是暂停状态;如果Activity是stop状态,这个Activity中所有的Fragment都不能被启动;如果Activity被销毁,那么它其中的所有Fragment都会被销毁。但是,当Activity在活动状态,可以独立控制Fragment的状态,比如加上或者移除Fragment。当这样进行fragment transaction(转换)的时候,可以把fragment放入Activity的back stack中,这样用户就可以进行返回操作。
方法 | 功能说明 |
---|---|
onAttach | 在Fragment与Activity关联之后被调用。虽然初始化Fragment参数可以通过getArguments()获得,但当Fragment附加到Activity之后,就无法再调用setArguments()。so除了在最开始时,其它时间都无法向初始化参数添加内容 |
onCreate | Fragment初次创建时调用。但此刻Activity还没有创建完成,因为我们的Fragment也是Activity创建的一部分。所以当你尝试在在这无法获取Activity的一些资源 |
onCreateView | 主要用于创建Fragment自身的UI,可用使用LayoutInflater对象的inflater()方法把布局xml映射为view,所以返回值必须是Fragment自身UI对应的view,但如果fragment没有用户界面可以返回null |
onActivityCreated | 在Activity的OnCreate()结束后,会调用此方法。即此时Fragment和Activity均已创建完毕。所以我们可以在这个方法里使用Activity的所有资源 |
onStart | 当Fragment对用户就是可见时,但用户还未开始与Fragment交互。因为是与Activity的OnStart()绑定的。写在Activity的OnStart()中处理的逻辑,换成用Fragment来实现时,依然可以放在OnStart()中来处理。 |
onResume | 当Fragment对用户可见并且正在运行时调用。即Fragment与用户交互之前的最后一个回调。与onStart类似,也是与Activity的OnResume是相互绑定的,它依赖于包含它的Activity的Activity.onResume。当OnResume()运行完毕之后,就可以正式与用户交互了 |
onPause | 与Activity的OnPause()绑定,意义也一样。当用户离开Fragment时第一个调用这个方法(但并不总意味着Fragment将被销毁)。按照经验,当用户结束会话之前,我们可以在这提交一些变化 |
onStop | 与Activity绑定,已停止的Fragment可以直接返回到OnStart()回调,然后调用OnResume()。 |
onDestroyView | 当Fragment将被结束或者保存,那么下一个回调将是onDestoryView(),将会删除在onCreateView创建的视图。下次这个Fragment若要显示,将会重新创建新视图。这会在onStop之后和onDestroy之前调用。经测试这个方法的调用同onCreateView是否返回非null视图无关。它会在的在这个视图状态被保存之后以及它被它的父Activity回收之前调用 |
onDestory | 当Fragment不再使用时被调用。但即使调用了onDestroy()阶段,仍可以从其依附的父Activity中找到,因为它还没有Detach。 |
onDetach | 当Fragment和Activity分离的时候调用,Fragment就不再与Activity相绑定,它也不再拥有视图层次结构,它的所有资源都将被释放 |
一旦Activity进入resumed状态(也就是running状态),我们就可以自由地添加和删除Fragment了。因此,只有当Activity在resumed状态时,Fragment的生命周期才能独立的运转,其它时候是依赖于Activity的生命周期变化的。
onAttach()——>onCreate()——>onCreateView()——>
onActivityCreated()——>
onStart()——>onResume()——>
onPause()——>onStop()——>
onPause()——>onStop()——>onDestroyView()——>onDestroy() ——>onDetach()——>
onCreate()——>onCreateView()——>onActivityCreated()——>
要使用Fragment,可通过继承Fragment类,复写相关方法来实现加载Fragment自身的UI并初始化Fragment相关变量,控制Fragment与Activity的交互,通常都需要指定一个UI,但也可以为Activity创建一个没有UI只提供后台行为的Fragment。
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical" android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView android:text="Hello Fragment!" android:layout_width="wrap_content"
android:layout_height="wrap_content"
style="@style/TitleStyle"/>
<fragment
android:id="@+id/id_fragment_main"
android:name="com.crazymo.fragmentsdemo.MainFragment"
android:layout_width="match_parent"
android:layout_height="wrap_content"
/>
LinearLayout>
package com.crazymo.fragmentsdemo;
import android.app.Fragment;
import android.content.Context;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
/**
* Created by cmo on 16-4-19.
*/
public class MainFragment extends Fragment {
private final String TAG="FramentDemo";
@Override
public void onAttach(Context context) {
Log.d(TAG, "onAttach:刚刚与Activity对接");
super.onAttach(context);
}
@Override
public void onCreate(Bundle savedInstanceState) {
Log.d(TAG, "onCreate:初始化Fragment对象");
super.onCreate(savedInstanceState);
}
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
Log.d(TAG, "onCreateView:初始化Fragment的UI");
return inflater.inflate(R.layout.fragment_main, container, false);//参数依次为:布局id,依附的容器
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
Log.d(TAG, "onActivityCreated:Activity和Fragment都已经创建好了");
super.onActivityCreated(savedInstanceState);
}
@Override
public void onStart() {
Log.d(TAG, "onStart:Fragment变为可见,待交互");
super.onStart();
}
@Override
public void onResume() {
Log.d(TAG, "onResume");
super.onResume();
}
@Override
public void onPause() {
Log.d(TAG, "onPause:与Activity的OnPause绑定");
super.onPause();
}
@Override
public void onStop() {
Log.d(TAG, "onStop:与Activity的OnStop绑定");
super.onStop();
}
@Override
public void onDestroyView() {
Log.d(TAG, "onDestroyView:Fragment即将被保存或者删除");
super.onDestroyView();
}
@Override
public void onDestroy() {
Log.d(TAG, "onDestroy:还与Activity藕断丝连中可以在Activity中找到");
super.onDestroy();
}
@Override
public void onDetach() {
Log.d(TAG, "onDetach:彻底和Activity分手了");
super.onDetach();
}
}
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical" android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
style="@style/TitleStyle"
android:text="Fragment详解(一)"/>
<ImageView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:src="@mipmap/ic_launcher"/>
LinearLayout>
return inflater.inflate(R.layout.fragment_main, container, false);//参数依次为:布局id,依附的容器
getFragmentManager().beginTransaction().add(R.id.id_fragment_main,new MainFragment()).commit();//第一个参数为容器的Id,这里为FrameLayout的Id,第二个参数为Fragment对象实例
getFragmentManager().beginTransaction().add(R.id.id_fragment_main,new MainFragment(),"MainFragment").commit();//第三个参数为tag
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:baselineAligned="false" >
<FrameLayout
android:id="@+id/id_fragment_main"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
LinearLayout>
FrgamentManager fragmentManager=getFragmentManager();
FragmentTransaction fragmentTransaction=fragmentManager.beginTransaction();
fragmentTrancsaction.add(id_fragment_main,new MainFragment());//容器的id,Fragment的实例对象
fragmentTransaction.commit();
Fragment的删除也还是借助FragmentManager和FragmentTransaction,主要有两个步骤:
Fragment的替换replace这个方法真的有点奇怪,并没有像我们正常的逻辑,至少应该提供源Fragment和目标Fragment两个参数,但并没有只是提供了一个容器Id和目标Fragment(即将被添加到容器里展示的Fragment),当年学习的时候也觉得很奇怪,这里先不深究,以后再去分析。
fragmentManager.beginTransaction().replace(R.id.id_fragment_main,new OtherFragment()).commit();//容器Id,新的Fragment对象
对于Fragment的hide或者show只需要传递我们一个将要hide或者show的Fragment即可。(需要注意的是无论是hide还是show都是针对顶层Fragment,也就是假设你要show的Fragment不是顶层的是show不出来的)
private void hideFragment(String tag){
Fragment mainFragment = fragmentManager.findFragmentByTag(tag);
fragmentManager.beginTransaction().hide(mainFragment).addToBackStack(tag).commit();
}
private void showFragment(String tag){
Fragment mainFragment = fragmentManager.findFragmentByTag(tag); fragmentManager.beginTransaction().show(mainFragment).addToBackStack(tag).commit();
}
fragmentManager.beginTransaction().attach(mainFragment).addToBackStack(tag).commit();
fragmentManager.beginTransaction().detach(mainFragment).addToBackStack(tag).commit();
主布局的xml文件:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical" android:layout_width="match_parent"
android:layout_height="match_parent">
<FrameLayout
android:id="@+id/id_fragment_main"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"
/>
<Button
android:id="@+id/id_addfragment_btn"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="添加Fragment1"/>
<Button
android:id="@+id/id_replacefragment_btn"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="替换为Fragment2"/>
<Button
android:id="@+id/id_removefragment_btn"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="移除Fragment1"/>
<Button
android:id="@+id/id_hidefragment_btn"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="hide MainFragement"/>
<Button
android:id="@+id/id_showfragment_btn"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="show MainFragement"/>
LinearLayout>
MainFragment的布局文件:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical" android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
style="@style/TitleStyle"
android:text="MainFragment"/>
<ImageView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:src="@mipmap/ic_launcher"/>
LinearLayout>
OtherFragment的布局文件:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical" android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
style="@style/TitleStyle"
android:text="OtherFragment"/>
<ImageView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:src="@mipmap/ic_blue_launcher"/>
LinearLayout>
MainActivity的代码:
package com.crazymo.fragmentsdemo;
import android.app.Activity;
import android.app.Fragment;
import android.app.FragmentManager;
import android.app.FragmentTransaction;
import android.nfc.Tag;
import android.support.annotation.NonNull;
import android.support.annotation.StringRes;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
public class MainActivity extends Activity implements View.OnClickListener {
private final String TAG="FramentDemo";
private Button mAddBtn,mRemoveBtn,mRelpaceBtn,mHideBtn,mShowBtn;
private FragmentManager fragmentManager=getFragmentManager();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d(TAG, "Activity onCreate");
setContentView(R.layout.activity_main);
init();
}
private void init(){
getViews();
setClickListener();
}
private void getViews(){
mAddBtn= (Button) findViewById(R.id.id_addfragment_btn);
mRelpaceBtn=(Button)findViewById(R.id.id_replacefragment_btn);
mRemoveBtn= (Button) findViewById(R.id.id_removefragment_btn);
mHideBtn= (Button) findViewById(R.id.id_hidefragment_btn);
mShowBtn= (Button) findViewById(R.id.id_showfragment_btn);
}
private void setClickListener(){
mAddBtn.setOnClickListener(this);
mRemoveBtn.setOnClickListener(this);
mRelpaceBtn.setOnClickListener(this);
mHideBtn.setOnClickListener(this);
mShowBtn.setOnClickListener(this);
}
@Override
protected void onStart() {
Log.d(TAG, "Activity onStart");
super.onStart();
}
@Override
protected void onRestart() {
Log.d(TAG,"Activity onReStart");
super.onRestart();
}
@Override
protected void onResume() {
Log.d(TAG,"Activity onResume");
super.onResume();
}
@Override
protected void onPause() {
Log.d(TAG,"Activity onPause");
super.onPause();
}
@Override
protected void onStop() {
Log.d(TAG,"Activity onStop");
super.onStop();
}
@Override
protected void onDestroy() {
Log.d(TAG,"Activity onDestory");
super.onDestroy();
}
private void addFragemnt(@StringRes int viewGroupId,@NonNull Fragment fragment,String tag){
fragmentManager=getFragmentManager();
fragmentManager.beginTransaction().add(viewGroupId,fragment,tag)
.addToBackStack(tag).commit();
}
private void removeFragment(String tag){
Fragment fragment = fragmentManager.findFragmentByTag(tag);
fragmentManager.beginTransaction().remove(fragment).commit();
}
private void replaceFragment(@StringRes int viewGroupId,@NonNull Fragment fragment){
fragmentManager.beginTransaction().add(viewGroupId, fragment).commit();
}
private void hideFragment(String tag){
Fragment mainFragment = fragmentManager.findFragmentByTag(tag);
fragmentManager.beginTransaction().hide(mainFragment).addToBackStack(tag).commit();
}
private void showFragment(String tag){
Fragment mainFragment = fragmentManager.findFragmentByTag(tag);
fragmentManager.beginTransaction().show(mainFragment).addToBackStack(tag).commit();
}
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.id_addfragment_btn:
int viewGroupId=R.id.id_fragment_main;
addFragemnt(viewGroupId,new MainFragment(),"MainFragment");
break;
case R.id.id_removefragment_btn:
removeFragment("MainFragment");
break;
case R.id.id_replacefragment_btn:
//Fragment fragment2 = fragmentManager.findFragmentByTag("MainFragment");
//fragmentManager.beginTransaction().remove(fragment2).commit();
int viewGroup=R.id.id_fragment_main;
replaceFragment(viewGroup,new OtherFragment());
break;
case R.id.id_hidefragment_btn:
hideFragment("MainFragment");
break;
case R.id.id_showfragment_btn:
showFragment("MainFragment");
break;
default:
break;
}
}
}
MainFragment的代码
package com.crazymo.fragmentsdemo;
import android.app.Fragment;
import android.content.Context;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
/**
* Created by cmo on 16-4-19.
*/
public class MainFragment extends Fragment {
private final String TAG="FramentDemo";
@Override
public void onAttach(Context context) {
Log.d(TAG, "onAttach:刚刚与Activity对接");
super.onAttach(context);
}
@Override
public void onCreate(Bundle savedInstanceState) {
Log.d(TAG, "onCreate:初始化Fragment对象");
super.onCreate(savedInstanceState);
}
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
Log.d(TAG, "onCreateView:初始化Fragment的UI");
return inflater.inflate(R.layout.fragment_main, container, false);
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
Log.d(TAG, "onActivityCreated:Activity和Fragment都已经创建好了");
super.onActivityCreated(savedInstanceState);
}
@Override
public void onStart() {
Log.d(TAG, "onStart:Fragment变为可见,待交互");
super.onStart();
}
@Override
public void onResume() {
Log.d(TAG, "onResume");
super.onResume();
}
@Override
public void onPause() {
Log.d(TAG, "onPause:与Activity的OnPause绑定");
super.onPause();
}
@Override
public void onStop() {
Log.d(TAG, "onStop:与Activity的OnStop绑定");
super.onStop();
}
@Override
public void onDestroyView() {
Log.d(TAG, "onDestroyView:Fragment即将被保存或者删除");
super.onDestroyView();
}
@Override
public void onDestroy() {
Log.d(TAG, "onDestroy:还与Activity藕断丝连中可以在Activity中找到");
super.onDestroy();
}
@Override
public void onDetach() {
Log.d(TAG, "onDetach:彻底和Activity分手了");
super.onDetach();
}
}
OtherFragment的代码
package com.crazymo.fragmentsdemo;
import android.app.Fragment;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
/**
* Created by cmo on 16-4-21.
*/
public class OtherFragment extends Fragment {
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
return inflater.inflate(R.layout.fragment_other,null);
}
}