Fragment知识介绍

来源:互联网 发布:淘宝商品购物车加不了 编辑:程序博客网 时间:2024/06/04 19:53

Fragment

看到这个词的第一反应肯定是什么是Fragment?

如上图,因为分辨率的问题产生原因:Fragment的出现就是为了解决不同设备之间分辨率的问题,避免了开发人员为了设备尺寸的大小而费心。Fragment代表了Activity的子模块,因此可以把Fragment理解成Activity片段。

1.Fragment的概述

Fragment的几个特征:

  • Fragment总是作为Activity界面的组成部分。Fragment可调用getActivity()方法获取它所在的Activity,
  • Activity可调用FragmentManager的findFragmentById()或findFragmentByTag()方法来获取Fragment。
  • 在Activity运行过程中,可调用FragmentManager的add()、remove()、replace()方法动态地添加、删除或替换Fragment。
    一个Activity可以同时组合多个Fragment;反过来,一个Fragment也可以被多个Activity复用。
  • Fragment可以响应自己的输入事件,并拥有自己的生命周期,但它们的生命周期直接被其所属的Activity的生命周期控制。

2.Fragment的生命周期

3.创建Fragment

与创建Activity类似,实现Fragment必须继承Fragment基类,Fragment继承体系图如下:

可以根据需要继承图中Fragment基类或它的任意子类。

接下来实现它的回调方法,例如:onCreate()、onCreateView()、onStart()、onResume()、onPause()、onStop()等等。

通常创建Fragment需要实现如下三个方法:

  • onCreate():系统创建Fragment对象后回调该方法,在实现代码中只初始化想要在Fragment中保持的必要组件。
  • onCreateView():当Fragment绘制界面组件时会回调该方法。该方法必须返回一个View,该View也就是该Fragment所显示的View。
  • onPause():当用户离开该Fragment时将回调该方法。

    下面我们根据书本来做一个简单的案例:
    BookDetailFragment代码

package com.text.allen.fra;import android.app.Fragment;import android.os.Bundle;import android.view.LayoutInflater;import android.view.View;import android.view.ViewGroup;import android.widget.TextView;/** * Created by lenovo on 2017/6/1. */public class BookDetailFragment extends Fragment {    public static final String ITEM_ID="item_id";    BookContent.Book book;    @Override    public void onCreate(Bundle savedInstanceState){        super.onCreate(savedInstanceState);        if(getArguments().containsKey(ITEM_ID)){            book=BookContent.ITEM_MAP.get(getArguments().getInt(ITEM_ID));        }    }    @Override    public View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState){        View rootView=inflater.inflate(R.layout.fragment_book_detail,container,false);        if(book!=null){            ((TextView)rootView.findViewById(R.id.book_title)).setText(book.title);            ((TextView)rootView.findViewById(R.id.book_desc)).setText(book.desc);        }        return  rootView;    }}

上面的Fragment将会加载并显示res/layout/目录下的fragment_book_detail.xml界面布局文件。①号代码获取启动该Fragment时传入的ITEM_ID参数,并根据该ID获取BookContent的ITEM_MAP中的图书信息。

BookContent类用于模拟系统的数据模型,该模拟类的代码如下:

package com.text.allen.fra;import java.util.ArrayList;import java.util.HashMap;import java.util.List;import java.util.Map;/** * Created by lenovo on 2017/6/2. */public class BookContent {    public static class Book{        public Integer id;        public String title;        public String desc;        public Book(Integer id,String title,String desc){            this.id=id;            this.title=title;            this.desc=desc;        }        @Override        public String toString(){            return title;        }    }    public static List<Book> ITEMS=new ArrayList<Book>();    public static Map<Integer,Book> ITEM_MAP=new HashMap<Integer, Book>();    static {        addItem(new Book(1, "Head First Java", "一本基础、入门的Java学习书籍,深受读者喜欢。"));        addItem(new Book(2, "Head First Android", "一本基础、入门的Android学习书籍,深受读者喜欢。"));        addItem(new Book(3, "Thinking in Java", "一本全面、深入的Java学习书籍,深受读者喜欢。"));    }    private static void addItem(Book book){        ITEMS.add(book);        ITEM_MAP.put(book.id,book);    }}

BookDetailFragment只是加载并显示一份简单的布局文件,这份布局文件(fragment_book_detail.xml)中通过LinearLayout包含两个文本框。该布局文件的代码如下:

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:orientation="vertical"    android:layout_width="match_parent"    android:layout_height="match_parent">    <TextView        style="?android:attr/textAppearance"        android:id="@+id/book_title"        android:padding="16dp"        android:layout_width="match_parent"        android:layout_height="wrap_content" />    <TextView        style="?android:attr/textAppearanceMedium"        android:padding="16dp"        android:id="@+id/book_desc"        android:layout_width="match_parent"        android:layout_height="wrap_content" /></LinearLayout>

创建ListFragment

如果开发ListFragment的子类,无须重写onCreateView()方法,只要调用ListFragment的setAdapter()方法为该Fragment设置Adapter即可。该ListFragment将会显示该Adapter提供的列表项。

public class BookListFragment extends ListFragment {    private Callbacks mCallbacks;    //定义一个回调接口,该Fragment所在Activity需要实现该接口    //该Fragment将通过该接口与它所在的Activity交互    public interface Callbacks {        public void onItemSelected(Integer id);    }    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        //为该ListFragment设置Adapter        setListAdapter(new ArrayAdapter<BookContent.Book>(getActivity(), android.R.layout.simple_list_item_activated_1, android.R.id.text1, BookContent.ITEMS));    }    //当该Fragment被添加、显示到Activity时,回调该方法    @Override    public void onAttach(Activity activity) {        super.onAttach(activity);        //如果Activity没有实现Callbacks接口,抛出异常        if (!(activity instanceof Callbacks)) {            throw new IllegalStateException("BookListFragment所在的Activity必须实现 Callbacks接口!");        }        //把该Activity当成Callbacks对象        mCallbacks = (Callbacks) activity;    }    //当该Fragment从它所属的Activity中被删除时回调该方法    @Override    public void onDetach() {        super.onDetach();        mCallbacks = null;    }    //当用户单击某列表时激发该回调方法    @Override    public void onListItemClick(ListView l, View v, int position, long id) {        super.onListItemClick(l, v, position, id);        //激发mCallbacks的onItemSelected方法        mCallbacks.onItemSelected(BookContent.ITEMS.get(position).id);    }    public void setActivateOnItemClick(boolean activateOnItemClick) {        getListView().setChoiceMode(activateOnItemClick ? ListView.CHOICE_MODE_SINGLE : ListView.CHOICE_MODE_NONE);    }}

为了控制ListFragment显示的列表项,只要调用ListFragment提供的setAdapter()方法,即可让该ListFragment显示该Adapter所提供的多个列表项。

Fragment与Activity通信

为了在Activity中显示Fragment,还必须将Fragment添加到Activity中。将Fragment添加到Activity中有如下两种方式:

  • 在布局文件中使用
<?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:layout_marginLeft="16dp"    android:layout_marginRight="16dp"    android:divider="?android:attr/dividerHorizontal"    android:orientation="horizontal"    android:showDividers="middle">    <fragment        android:id="@+id/book_list"        android:name="com.trampcr.bookfragmentdemo.BookListFragment"        android:layout_width="0dp"        android:layout_height="match_parent"        android:layout_weight="1" />    <FrameLayout        android:id="@+id/book_detail_container"        android:layout_width="0dp"        android:layout_height="match_parent"        android:layout_weight="3"/></LinearLayout>

上面布局文件中使用

public class MainActivity extends AppCompatActivity implements BookListFragment.Callbacks{    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_book_twopane);    }    @Override    public void onItemSelected(Integer id) {        //创建Bundle,准备向Fragment传入参数        Bundle arguments = new Bundle();        arguments.putInt(BookDetailFragment.ITEM_ID, id);        //创建BookDetailFragment对象        BookDetailFragment fragment = new BookDetailFragment();        //向Fragment传入参数        fragment.setArguments(arguments);        //使用fragment替换book_detail_container容器当前显示的Fragment        getFragmentManager().beginTransaction().replace(R.id.book_detail_container, fragment).commit();//①    }}

上面的①号代码调用了FragmentTransaction的replace()方法动态更新了ID为book_detail_container容器中显示的Fragment。
Fragment与Activity相互传递数据的方式:

  • Activity向Fragment传递数据:在Activity中创建Bundle数据包,并调用Fragment的setArgument(Bundle
    bundle)方法即可将Bundle数据包传给Fragment。
  • Fragment向Activity传递数据或Activity需要在Fragment运行中进行实时通信:在Fragment中定义一个内部回调接口,再让包含该Fragment的Activity实现该回调接口,这样Fragment即可调用该回调方法将数据传给Activity。

上面示例定义了两个Fragment,并使用一个Activity来组合这两个Fragment,使用大屏幕设备运行该程序,可以看到:
这里写图片描述

作者:王宇:原文地址

原创粉丝点击