http://www.cnblogs.com/mengdd/archive/2013/01/08/2851368.html

Fragment

  Android是在Android 3.0 (API level 11)开始引入Fragment的。

  可以把Fragment想成Activity中的模块,这个模块有自己的布局,有自己的生命周期,单独处理自己的输入,在Activity运行的时候可以加载或者移除Fragment模块。

  可以把Fragment设计成可以在多个Activity中复用的模块。

  当开发的应用程序同时适用于平板电脑和手机时,可以利用Fragment实现灵活的布局,改善用户体验。

  如图:

Fragment的生命周期

  因为Fragment必须嵌入在Acitivity中使用,所以Fragment的生命周期和它所在的Activity是密切相关的。

  如果Activity是暂停状态,其中所有的Fragment都是暂停状态;如果Activity是stopped状态,这个Activity中所有的Fragment都不能被启动;如果Activity被销毁,那么它其中的所有Fragment都会被销毁。

  但是,当Activity在活动状态,可以独立控制Fragment的状态,比如加上或者移除Fragment。

  当这样进行fragment transaction(转换)的时候,可以把fragment放入Activity的back stack中,这样用户就可以进行返回操作。

Fragment的使用相关

  使用Fragment时,需要继承Fragment或者Fragment的子类(DialogFragment, ListFragment, PreferenceFragment, WebViewFragment),所以Fragment的代码看起来和Activity的类似。

使用Support Library

  Support Library是一个提供了API库函数的JAR文件,这样就可以在旧版本的Android上使用一些新版本的APIs。

  比如android-support-v4.jar.它的完整路径是:

  <sdk>/extras/android/support/v4/android-support-v4.jar.

  它就提供了Fragment的APIs,使得在Android 1.6 (API level 4)以上的系统都可以使用Fragment。

  为了确定没有在旧版本系统上使用新版本的APIs,需要如下导入语句:

  import android.support.v4.app.Fragment;   import android.support.v4.app.FragmentManager;

  同时应该将上述的包拷入libs项目下的libs文件夹,然后在项目的Properties中添加:右键单击项目,选Properties,左边选Java Build Path,然后Add External JARs…,添加android-support-v4.jar.

  当创建包含Fragment的Activity时,如果用的是Support Library,那么继承的就应该是FragmentActivity而不是Activity。

必须实现的三个回调函数

  onCreate()

  系统在创建Fragment的时候调用这个方法,这里应该初始化相关的组件,一些即便是被暂停或者被停止时依然需要保留的东西。

  onCreateView()

  当第一次绘制Fragment的UI时系统调用这个方法,必须返回一个View,如果Fragment不提供UI也可以返回null。

  注意,如果继承自ListFragment,onCreateView()默认的实现会返回一个ListView,所以不用自己实现。

  onPause()

  当用户离开Fragment时第一个调用这个方法,需要提交一些变化,因为用户很可能不再返回来。

实现Fragment的UI

  提供Fragment的UI,必须实现onCreateView()方法。

  假设Fragment的布局设置写在example_fragment.xml资源文件中,那么onCreateView()方法可以如下写:

  

public static class ExampleFragment extends Fragment { @Override public View onCreateView(LayoutInflater inflater, ViewGroup container,   Bundle savedInstanceState) { // Inflate the layout for this fragment return inflater.inflate(R.layout.example_fragment, container, false); } }

  onCreateView()中container参数代表该Fragment在Activity中的父控件;savedInstanceState提供了上一个实例的数据。

  inflate()方法的三个参数:

  第一个是resource ID,指明了当前的Fragment对应的资源文件;

  第二个参数是父容器控件;

  第三个布尔值参数表明是否连接该布局和其父容器控件,在这里的情况设置为false,因为系统已经插入了这个布局到父控件,设置为true将会产生多余的一个View Group。

把Fragment加入Activity

  当Fragment被加入Activity中时,它会处在对应的View Group中。

  Fragment有两种加载方式:一种是在Activity的layout中使用标签<fragment>声明;另一种方法是在代码中把它加入到一个指定的ViewGroup中。

  另外,Fragment它可以并不是Activity布局中的任何一部分,它可以是一个不可见的部分。这部分内容先略过。

加载方式1:通过Activity的布局文件将Fragment加入Activity

  在Activity的布局文件中,将Fragment作为一个子标签加入即可。

  如:

<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="horizontal" android:layout_width="match_parent" android:layout_height="match_parent"> <fragment android:name="com.example.news.ArticleListFragment" android:id="@+id/list" android:layout_weight="1" android:layout_width="0dp" android:layout_height="match_parent" /> <fragment android:name="com.example.news.ArticleReaderFragment" android:id="@+id/viewer" android:layout_weight="2" android:layout_width="0dp" android:layout_height="match_parent" /> </LinearLayout>

  其中android:name属性填上你自己创建的fragment的完整类名。

  当系统创建这个Activity的布局文件时,系统会实例化每一个fragment,并且调用它们的onCreateView()方法,来获得相应fragment的布局,并将返回值插入fragment标签所在的地方。

  有三种方法为Fragment提供ID:

  android:id属性:唯一的id

  android:tag属性:唯一的字符串

  如果上面两个都没提供,系统使用容器view的ID。

加载方式2:通过编程的方式将Fragment加入到一个ViewGroup中

  当Activity处于Running状态下的时候,可以在Activity的布局中动态地加入Fragment,只需要指定加入这个Fragment的父View Group即可。

  首先,需要一个FragmentTransaction实例: 

FragmentManager fragmentManager = getFragmentManager()FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();

  (注,如果import android.support.v4.app.FragmentManager;那么使用的是:FragmentManager fragmentManager = getSupportFragmentManager();)

  

  之后,用add()方法加上Fragment的对象:

ExampleFragment fragment = new ExampleFragment(); fragmentTransaction.add(R.id.fragment_container, fragment); fragmentTransaction.commit();

  其中第一个参数是这个fragment的容器,即父控件组。

  最后需要调用commit()方法使得FragmentTransaction实例的改变生效。

实例

两个Fragment类

package com.example.android.fragments;import android.support.v4.app.Fragment;import android.os.Bundle;import android.view.LayoutInflater;import android.view.View;import android.view.ViewGroup;import android.widget.TextView;public class ArticleFragment extends Fragment {    final static String ARG_POSITION = "position";    int mCurrentPosition = -1;    @Override    public View onCreateView(LayoutInflater inflater, ViewGroup container,         Bundle savedInstanceState) {        // If activity recreated (such as from screen rotate), restore        // the previous article selection set by onSaveInstanceState().        // This is primarily necessary when in the two-pane layout.        if (savedInstanceState != null) {            mCurrentPosition = savedInstanceState.getInt(ARG_POSITION);        }        // Inflate the layout for this fragment        return inflater.inflate(R.layout.article_view, container, false);    }    @Override    public void onStart() {        super.onStart();        // During startup, check if there are arguments passed to the fragment.        // onStart is a good place to do this because the layout has already been        // applied to the fragment at this point so we can safely call the method        // below that sets the article text.        Bundle args = getArguments();        if (args != null) {            // Set article based on argument passed in            updateArticleView(args.getInt(ARG_POSITION));        } else if (mCurrentPosition != -1) {            // Set article based on saved instance state defined during onCreateView            updateArticleView(mCurrentPosition);        }    }    public void updateArticleView(int position) {        TextView article = (TextView) getActivity().findViewById(R.id.article);        article.setText(Ipsum.Articles[position]);        mCurrentPosition = position;    }    @Override    public void onSaveInstanceState(Bundle outState) {        super.onSaveInstanceState(outState);        // Save the current article selection in case we need to recreate the fragment        outState.putInt(ARG_POSITION, mCurrentPosition);    }}

package com.example.android.fragments;import android.app.Activity;import android.os.Build;import android.os.Bundle;import android.support.v4.app.ListFragment;import android.view.View;import android.widget.ArrayAdapter;import android.widget.ListView;public class HeadlinesFragment extends ListFragment {    OnHeadlineSelectedListener mCallback;    // The container Activity must implement this interface so the frag can deliver messages    public interface OnHeadlineSelectedListener {        /** Called by HeadlinesFragment when a list item is selected */        public void onArticleSelected(int position);    }    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        // We need to use a different list item layout for devices older than Honeycomb        int layout = Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB ?                android.R.layout.simple_list_item_activated_1 : android.R.layout.simple_list_item_1;        // Create an array adapter for the list view, using the Ipsum headlines array        setListAdapter(new ArrayAdapter<String>(getActivity(), layout, Ipsum.Headlines));    }    @Override    public void onStart() {        super.onStart();        // When in two-pane layout, set the listview to highlight the selected list item        // (We do this during onStart because at the point the listview is available.)        if (getFragmentManager().findFragmentById(R.id.article_fragment) != null) {            getListView().setChoiceMode(ListView.CHOICE_MODE_SINGLE);        }    }    @Override    public void onAttach(Activity activity) {        super.onAttach(activity);        // This makes sure that the container activity has implemented        // the callback interface. If not, it throws an exception.        try {            mCallback = (OnHeadlineSelectedListener) activity;        } catch (ClassCastException e) {            throw new ClassCastException(activity.toString()                    + " must implement OnHeadlineSelectedListener");        }    }    @Override    public void onListItemClick(ListView l, View v, int position, long id) {        // Notify the parent activity of selected item        mCallback.onArticleSelected(position);                // Set the item as checked to be highlighted when in two-pane layout        getListView().setItemChecked(position, true);    }}

主Activity

package com.example.android.fragments;import android.os.Bundle;import android.support.v4.app.FragmentActivity;import android.support.v4.app.FragmentTransaction;public class MainActivity extends FragmentActivity         implements HeadlinesFragment.OnHeadlineSelectedListener {    /** Called when the activity is first created. */    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.news_articles);        // Check whether the activity is using the layout version with        // the fragment_container FrameLayout. If so, we must add the first fragment        if (findViewById(R.id.fragment_container) != null) {            // However, if we're being restored from a previous state,            // then we don't need to do anything and should return or else            // we could end up with overlapping fragments.            if (savedInstanceState != null) {                return;            }            // Create an instance of ExampleFragment            HeadlinesFragment firstFragment = new HeadlinesFragment();            // In case this activity was started with special instructions from an Intent,            // pass the Intent's extras to the fragment as arguments            firstFragment.setArguments(getIntent().getExtras());            // Add the fragment to the 'fragment_container' FrameLayout            getSupportFragmentManager().beginTransaction()                    .add(R.id.fragment_container, firstFragment).commit();        }    }    public void onArticleSelected(int position) {        // The user selected the headline of an article from the HeadlinesFragment        // Capture the article fragment from the activity layout        ArticleFragment articleFrag = (ArticleFragment)                getSupportFragmentManager().findFragmentById(R.id.article_fragment);        if (articleFrag != null) {            // If article frag is available, we're in two-pane layout...            // Call a method in the ArticleFragment to update its content            articleFrag.updateArticleView(position);        } else {            // If the frag is not available, we're in the one-pane layout and must swap frags...            // Create fragment and give it an argument for the selected article            ArticleFragment newFragment = new ArticleFragment();            Bundle args = new Bundle();            args.putInt(ArticleFragment.ARG_POSITION, position);            newFragment.setArguments(args);            FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();            // Replace whatever is in the fragment_container view with this fragment,            // and add the transaction to the back stack so the user can navigate back            transaction.replace(R.id.fragment_container, newFragment);            transaction.addToBackStack(null);            // Commit the transaction            transaction.commit();        }    }}

首页布局文件,分是否large平板模式

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:id="@+id/fragment_container"    android:layout_width="match_parent"    android:layout_height="match_parent" />

large平板

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:orientation="horizontal"    android:layout_width="match_parent"    android:layout_height="match_parent">    <fragment android:name="com.example.android.fragments.HeadlinesFragment"              android:id="@+id/headlines_fragment"              android:layout_weight="1"              android:layout_width="0dp"              android:layout_height="match_parent" />    <fragment android:name="com.example.android.fragments.ArticleFragment"              android:id="@+id/article_fragment"              android:layout_weight="2"              android:layout_width="0dp"              android:layout_height="match_parent" /></LinearLayout>

运用的知识,fragment间通信

1、使用接口

2、重载onAttach方法

public void onAttach(Activity activity)

获取容器的Activity

 

更多相关文章

  1. Android(安卓)WebView 踩过的坑
  2. Android(安卓)N 各种ANR的时间
  3. Android(安卓)NavigationView使用详解
  4. android-PullToRefreshListView自动刷新问题
  5. Android(安卓)常用代码集合
  6. Android(安卓)studio配置Kotlin
  7. 解决android studio国内开发者Gradle下载慢问题
  8. Android(安卓)Device Chooser中显示Target unknown解决方法
  9. ViewStub must have a valid layoutResource

随机推荐

  1. Ubuntu14.04 下面安装SVN服务器
  2. Ubuntu 12.04 64位 使用 Android(安卓)问
  3. 为什么listview的item变黑了
  4. android环境搭建——工欲善其事,必先利其
  5. android developer API Guider之user int
  6. 游戏开发基本观
  7. 传统menu的使用方法总结
  8. Android(安卓)Intent 其中一个分析
  9. Android开源项目xUtils HttpUtils模块分
  10. Android(安卓)基础-Activity的使用(二)