1ListFragment
今天首先学习了一种非常常用的展示场景:列表展示。 
昨天学习了使用Fragmet来代替activity进行设计,今天在托管单个fragment的基础上,掌握托管一个布局list。先看下效果:
因为Fragment列表需要使用ArrayList保存,而为了使Fragment对象受acrivity等生命周期影响, 创建如下单例类:
public class CrimeLab {
    private ArrayList<Crime> mCrimes;
    private static CrimeLab sCrimeLab;
    private  Context mAppContext;
    private CrimeLab(Context context){
        mAppContext=context;
        mCrimes=new ArrayList<Crime>();
        for(int i=0;i<100;i++){
            Crime c =new Crime();
            c.setTitle("Crime #"+i);
            c.setSolved(i%2==0);
            mCrimes.add(c);
        }
    }
    public static CrimeLab get(Context c){
        if(sCrimeLab==null){
            sCrimeLab=new CrimeLab(c.getApplicationContext());
        }
        return sCrimeLab;
    }
    public ArrayList<Crime> getmCrimes(){
        return mCrimes;
    }
    public Crime getCrime(UUID id){
        for (Crime c: mCrimes){
            if(c.getId().equals(id)){
                return c;
            }
        }
        return null;
    }
}
同时,还要建立相应的fragment布局文件和类文件,实际上,存在专门支持list fragment的类:ListFragment。继承这个类即可,之后可使用其内置的listadapter。
为什么使用adapter呢? 因为我们的fragmentlab中新建了100个fragment对象,而我们不可能在一个页面中把他们全部显示出来,而是需要显示时才创建对象。adapter就是从模型层获得数据,并把它提供给ListView显示的桥梁。
private class CrimeAdapter extends ArrayAdapter<Crime>{
                public CrimeAdapter(ArrayList<Crime> crimes){
                    super(getActivity(),0,crimes);
                }
                @Override
                public View getView(int position,View convertView,ViewGroup parent){
                    if(convertView==null){
                        convertView=getActivity().getLayoutInflater().inflate(R.layout.list_item_crime,null);
                    }
                    Crime c=getItem(position);
                    TextView titleTextView=(TextView)convertView.findViewById(R.id.crime_list_item_titleTextView);
                    titleTextView.setText(c.getTitle());
                    TextView dateTextView=(TextView)convertView.findViewById(R.id.crime_list_item_dateTextView);
                    dateTextView.setText(c.getDate().toString());
                    CheckBox solvedCheckBox=(CheckBox)convertView.findViewById(R.id.crime_list_item_solvedCheckBox);
                    solvedCheckBox.setChecked(c.isSolved());
                    return convertView;
        }
    }
实现自己定制Adapater的代码,之所以要实现自己的adapter,是因为在list的每一个条目我们定制了自己的布局。如代码中的list_item_crime.
这样就得到了list显示的fragment。
2ViewPager
ViewPager可以实现左右滑动屏幕切换查看不同列表项的功能。
ViewPager需要借助于Adapter才能够提供视图。 通过PagerAdapter的子类:FragmentStatePagerAdapter来处理两者间的配合问题。
这里需要实现两个方法,getCount()和getItem()。代码示例如下: 
mViewPager.setAdapter(new FragmentStatePagerAdapter(fm) {
            @Override
            public int getCount(){
                return mCrimes.size();
            }
            @Override
            public Fragment getItem(int pos) {
                Crime crime=mCrimes.get(pos);
                return CrimeFragment.newInstance(crime.getId());
            }
        });
3 fragment数据传输
跟activity类似,fragment间也可以进行数据传输。而且fragment级的数据传输会使编程更加灵活。
试想如下场景: 在CrimeFragment中需要按键调出DatePickerFragment, 后者的初始化需要前者提供的数据。同时,DatePickerFragment的返回值也要作用于CrimeFragment。
需要做的事情有如下几步:
1)从CrimeFragment 初始化DatePickerFragment时, 将数据作为构造参数传入
2)DatePickerFragment构造时,将传入的值保存到argument
3)DatePickerFragment渲染时,取arguments中值
4)DatePickerFragment将值回传给CrimeFragment  
为实现以上过程,首先要在DatePickerFragment 编写newInstance方法, 改方法可以在实例化DatePickerFragment时被调用并接受参数,同时在fragment create前准备好数据 
public static DatePickerFragment newInstance(Date date){
        Bundle args=new Bundle();
        args.putSerializable(EXTRA_DATE,date);
        DatePickerFragment fragment=new DatePickerFragment();
        fragment.setArguments(args);
        return fragment;
    }
返回数据时,则覆盖onActivityResult方法。
交互过程如: