Android 管理Fragments

管理Fragments

FragmentManager

  为了管理Activity中的fragments,需要使用FragmentManager.

  为了得到它,需要调用Activity中的getFragmentManager()方法。

  因为FragmentManager的API是在Android 3.0,也即API level 11开始引入的,所以对于之前的版本,需要使用support library中的FragmentActivity,并且使用getSupportFragmentManager()方法。

 

用FragmentManager可以做的工作有:

  得到Activity中存在的fragment:

  使用findFragmentById()或findFragmentByTag()方法。

  将fragment弹出back stack:

  popBackStack():将back stack中最后一次的fragment转换弹出。如果没有可以出栈的东西,返回false。

  这个函数是异步的:它将弹出栈的请求加入队列,但是这个动作直到应用回到事件循环才会执行。

  为back stack加上监听器:

  addOnBackStackChangedListener()

 

Performing Fragment Transactions

  使用Fragment时,可以通过用户交互来执行一些动作,比如增加、移除、替换等。

  所有这些改变构成一个集合,这个集合被叫做一个transaction。

  可以调用FragmentTransaction中的方法来处理这个transaction,并且可以将transaction存进由activity管理的back stack中,这样用户就可以进行fragment变化的回退操作。

  可以这样得到FragmentTransaction类的实例: 

FragmentManager fragmentManager = getFragmentManager();



FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();

 

  每个transaction是一组同时执行的变化的集合。

  用add(), remove(), replace()方法,把所有需要的变化加进去,然后调用commit()方法,将这些变化应用。

  在commit()方法之前,你可以调用addToBackStack(),把这个transaction加入back stack中去,这个back stack是由activity管理的,当用户按返回键时,就会回到上一个fragment的状态。

  比如下面的代码就是用一个新的fragment取代之前的fragment,并且将前次的状态存储在back stack中。

  

// Create new fragment and transaction

Fragment newFragment = new ExampleFragment();

FragmentTransaction transaction = getFragmentManager().beginTransaction();



// Replace whatever is in the fragment_container view with this fragment,

// and add the transaction to the back stack

transaction.replace(R.id.fragment_container, newFragment);

transaction.addToBackStack(null);



// Commit the transaction

transaction.commit();

 

  在这个例子中,newFragment将取代在R.id.fragment_container容器中的fragment,如果没有,将直接添加新的fragment。

  通过调用addToBackStack(),commit()的一系列转换作为一个transaction被存储在back stack中,用户按Back键可以返回上一个转换前的状态。

  当你移除一个fragment的时候,如果commit()之前没有调用addToBackStack(),那个fragment将会是destroyed;如果调用了addToBackStack(),这个fragment会是stopped,可以通过返回键来恢复。

 

关于commit()方法

  调用commit()方法并不能立即执行transaction中包含的改变动作,commit()方法把transaction加入activity的UI线程队列中。

  但是,如果觉得有必要的话,可以调用executePendingTransactions()方法来立即执行commit()提供的transaction。

  这样做通常是没有必要的,除非这个transaction被其他线程依赖。

  注意:你只能在activity存储它的状态(当用户要离开activity时)之前调用commit(),如果在存储状态之后调用commit(),将会抛出一个异常

  这是因为当activity再次被恢复时commit之后的状态将丢失。如果丢失也没关系,那么使用commitAllowingStateLoss()方法。

 

实例程序

  写了个小程序实践了一下fragment的管理,程序不是很完善,就是试试基本用法,先按第一个按钮添加一个fragment,第二个按钮将其替换,第三个按钮将第二个按钮添加的fragment删除。

  相关代码:

  第一个fragment:

Android 管理Fragments ExampleFragment.java
package com.example.learningfragment;



import android.os.Bundle;

import android.support.v4.app.Fragment;

import android.view.LayoutInflater;

import android.view.View;

import android.view.ViewGroup;







public class ExampleFragment extends Fragment

{



    //三个一般必须重载的方法

    @Override

    public void onCreate(Bundle savedInstanceState)

    {

        // TODO Auto-generated method stub

        super.onCreate(savedInstanceState);

        System.out.println("ExampleFragment--onCreate");

    }



    @Override

    public View onCreateView(LayoutInflater inflater, ViewGroup container,

            Bundle savedInstanceState)

    {

        System.out.println("ExampleFragment--onCreateView");

        return inflater.inflate(R.layout.example_fragment_layout, container, false);

        

    }



    @Override

    public void onPause()

    {

        // TODO Auto-generated method stub

        super.onPause();

        System.out.println("ExampleFragment--onPause");

    }



    



    @Override

    public void onResume()

    {

        // TODO Auto-generated method stub

        super.onResume();

        System.out.println("ExampleFragment--onResume");

    }



    @Override

    public void onStop()

    {

        // TODO Auto-generated method stub

        super.onStop();

        System.out.println("ExampleFragment--onStop");

    }

    

    



}

  它的布局:

Android 管理Fragments example_fragment_layout.xml
<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

    android:layout_width="match_parent"

    android:layout_height="match_parent"

    android:orientation="vertical" >

    <TextView

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:gravity="left"

        android:textSize="20dip"

        android:text="@string/fragment1"

           />

    <TextView

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:text="@string/num1"

           />

    <TextView

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:text="@string/num2"

        />  

</LinearLayout>


  第二个fragment:

Android 管理Fragments NewFragment.java
package com.example.learningfragment;



import android.os.Bundle;

import android.support.v4.app.Fragment;

import android.view.LayoutInflater;

import android.view.View;

import android.view.ViewGroup;



public class NewFragment extends Fragment

{



    //三个一般必须重载的方法

    @Override

    public void onCreate(Bundle savedInstanceState)

    {

        // TODO Auto-generated method stub

        super.onCreate(savedInstanceState);

        System.out.println("NewFragment--onCreate");

    }



    @Override

    public View onCreateView(LayoutInflater inflater, ViewGroup container,

            Bundle savedInstanceState)

    {

        System.out.println("NewFragment--onCreateView");

        return inflater.inflate(R.layout.new_fragment_layout, container, false);

        

    }



    @Override

    public void onPause()

    {

        // TODO Auto-generated method stub

        super.onPause();

        System.out.println("NewFragment--onPause");

    }



}
Android 管理Fragments new_fragment_layout.xml
<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

    android:layout_width="match_parent"

    android:layout_height="match_parent"

    android:orientation="vertical" > 

    <TextView

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:textSize="20dip"

        android:gravity="left"

        android:text="@string/fragment2"

           />  

    <TextView

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:text="@string/num3"

           />

    <TextView

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:text="@string/num4"

        />  

</LinearLayout>

  主Activity:

Android 管理Fragments LearnFragment.java
package com.example.learningfragment;



import android.os.Bundle;

import android.support.v4.app.FragmentActivity;

import android.support.v4.app.FragmentManager;

import android.support.v4.app.FragmentTransaction;

import android.view.View;

import android.widget.Button;





public class LearnFragment extends FragmentActivity

{

    Button btn1;

    Button btn2;

    Button btn3;

    ExampleFragment fragmentE;

    NewFragment fragmentN;

    FragmentManager fragmentManager;

    

    @Override

    public void onCreate(Bundle savedInstanceState)

    {

        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_learn_fragment);

        findViews();

        setListeners();

        

        //获得Fragment管理所需要的类的对象

        fragmentManager = getSupportFragmentManager();

        



    }

    

    private void findViews()

    {

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

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

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

        

    }

    

    private void setListeners()

    {

        //第一个按钮,增加一个ExampleFragment

        btn1.setOnClickListener(new Button.OnClickListener()

        {



            public void onClick(View v)

            {    

                

                //在程序中加入ExampleFragment                

                fragmentE = new ExampleFragment();



                FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();

                fragmentTransaction.add(R.id.linear1,fragmentE);

                fragmentTransaction.addToBackStack(null);            

                fragmentTransaction.commit();                

            }

            

        }

        );

        

        //第二个按钮,用一个NewFragment替换前面增加的那个fragment

        btn2.setOnClickListener(new Button.OnClickListener()

        {



            public void onClick(View v)

            {                

                fragmentN = new NewFragment();

                FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();

                fragmentTransaction.replace(R.id.linear1,fragmentN);

                fragmentTransaction.addToBackStack(null);

                fragmentTransaction.commit();

                

            }

            

        }

        );

        

        //第三个按钮,移除fragment

        btn3.setOnClickListener(new Button.OnClickListener()

        {



            public void onClick(View v)

            {



                FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();

                fragmentTransaction.remove(fragmentN);

                fragmentTransaction.addToBackStack(null);

                fragmentTransaction.commit();

                

            }

            

        }

        );

    }



}
Android 管理Fragments activity_learn_fragment.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

    xmlns:tools="http://schemas.android.com/tools"

    android:id="@+id/linear1"

    android:layout_width="match_parent"

    android:layout_height="match_parent"

    android:orientation="vertical"

    >

    <TextView

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:textSize="20dip"

        android:gravity="center_horizontal"

        android:text="@string/layout1"

           />

    <Button

        android:id="@+id/btn1"

        android:layout_width="fill_parent"

        android:layout_height="wrap_content"

        android:text="@string/btn1"      

        />



    <fragment 

        android:name="com.example.learningfragment.ExampleFragment"

        android:id="@+id/fragment1"

        android:layout_width="match_parent"

        android:layout_height="wrap_content"



        />

    <Button

        android:id="@+id/btn2"

        android:layout_width="fill_parent"

        android:layout_height="wrap_content"

        android:text="@string/btn2"      

        />



    <LinearLayout

        xmlns:android="http://schemas.android.com/apk/res/android"

        android:id="@+id/linear2"

        android:layout_width="fill_parent"

        android:layout_height="wrap_content"

        android:orientation="vertical"

        >

    <TextView

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:textSize="20dip"

        android:gravity="center_horizontal"

        android:text="@string/layout2"

           />

     <Button

        android:id="@+id/btn3"

        android:layout_width="fill_parent"

        android:layout_height="wrap_content"

        android:text="@string/btn3"      

        />

     </LinearLayout>



</LinearLayout>

  资源:

Android 管理Fragments strings.xml
<resources>



    <string name="app_name">LearningFragment</string>

    <string name="hello_world">Hello world!</string>

    <string name="menu_settings">Settings</string>

    <string name="title_activity_learn_fragment">LearnFragment</string>



    <string name="layout1">LinearLayout1</string>

    <string name="layout2">LinearLayout2</string>

    

    <string name="fragment1">FragmentType1</string>

    <string name="fragment2">FragmentType2</string>

    

    <string name="num1">NO.1</string>

    <string name="num2">NO.2</string>

    <string name="num3">NO.3</string>

    <string name="num4">NO.4</string>

    

    <string name="btn1">Add fragment</string>

    <string name="btn2">Replace fragment</string>

    <string name="btn3">Remove fragment</string>



</resources>

 

  程序运行截图:

Android 管理Fragments

参考资料:

  API Guides:Fragments

  http://developer.android.com/guide/components/fragments.html

  FragmentManager类文档:

  http://developer.android.com/reference/android/app/FragmentManager.html

  FragmentTransaction类文档

  http://developer.android.com/reference/android/app/FragmentTransaction.html

 

 

 

你可能感兴趣的:(Fragment)