您的位置:首页 > 移动开发 > Android开发

新人培训-Fragment的使用

2017-11-29 10:50 176 查看

mindLine



本节课用到的样例在我的github欢迎关注��

我们为什么要用Fragment

简单的讲,Fragment有一些类似于activity的控件,它主要有以下几个优点:

* 便于各种屏幕的适应,可以为不同屏幕大小的设备创建动态灵活的UI(小屏幕手机到大屏幕平板电脑),极大提高了UI的灵活性。





利于代码复用,每个frgment都是独立的,可以被多个activity公用。

1.Fragment的生命周期



onAttach()

作用:fragment已经关联到activity,

这个是 回调函数
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
Log.i("onAttach_Fragment");
}
这个时候 activity已经传进来了
获得activity的传递的值
就可以进行 与activity的通信里

当然也可以使用getActivity(),前提是这个fragment已经和宿主的activity关联,并且没有脱离
他只调用一次。


onCreate()

系统创建fragment的时候回调他,在他里面实例化一些变量

这些个变量主要是:当你 暂停 停止的时候 你想保持的数据

如果我们要为fragment启动一个后台线程,可以考虑将代码放于此处。

参数是:Bundle savedInstance, 用于保存 Fragment 参数, Fragement 也可以 重写 onSaveInstanceState(BundleoutState) 方法, 保存Fragement状态;

可以用于 文件保护

他只调用一次。

onCreateView()

第一次使用的时候 fragment会在这上面画一个layout出来,
为了可以画控件 要返回一个 布局的view,也可以返回null

当系统用到fragment的时候 fragment就要返回他的view,越快越好,所以尽量在这里不要做耗时操作,比如从数据库加载大量数据显示listview,当然线程还是可以的。

给当前的fragment绘制ui布局,可以使用线程更新UI
说白了就是加载fragment的布局的。
这里一般都先判断是否为null


if(text==null){

Bundle args=getArguments();

text=args.getString(“text”);

}

if (view == null) {

view = inflater.inflate(R.layout.hello, null);

}

这样进行各判断省得每次都要加载,减少资源消耗

onActivityCreated()

当Activity中的onCreate方法执行完后调用。

注意了:
从这句官方的话可以看出:当执行onActivityCreated()的时候 activity的onCreate才刚完成。所以在onActivityCreated()调用之前 activity的onCreate可能还没有完成,所以不能再onCreateView()中进行 与activity有交互的UI操作,UI交互操作可以砸onActivityCreated()里面进行。
所以呢:这个方法主要是初始化那些你需要你的父Activity或者Fragment的UI已经被完整初始化才能初始化的元素。
如果在onCreateView里面初始化空间 会慢很多,比如listview等


onStart()

和activity一致 启动, Fragement 启动时回调, 此时Fragement可见;


onResume()

和activity一致  在activity中运行是可见的
激活, Fragement 进入前台, 可获取焦点时激活;


onPause()

和activity一致  其他的activity获得焦点,这个仍然可见
第一次调用的时候,指的是 用户 离开这个fragment(并不是被销毁)
通常用于 用户的提交(可能用户离开后不会回来了)


onStop()

和activity一致
fragment不可见的, 可能情况:activity被stopped了 OR  fragment被移除但被加入到回退栈中
一个stopped的fragment仍然是活着的如果长时间不用也会被移除


onDestroyView()

Fragment中的布局被移除时调用。
表示fragemnt销毁相关联的UI布局
清除所有跟视图相关的资源

以前以为这里没什么用处其实 大有文章可做,
相信大家都用过ViewPager+Fragment,由于ViewPager的缓存机制,每次都会加载3页。
例如:有四个 fragment 当滑动到第四页的时候 第一页执行onDestroyView(),但没有执行onDestroy。他依然和activity关联。当在滑动到第一页的时候又执行了 onCreateView()。 生命周期可以自己试一下。
那么问题来了。会出现重复加载view的局面,所以这么做(下面是先人的代码)


@Override

public void onDestroyView() {

Log.i(“onDestroyView_Fragment”);

if(view!=null){

((ViewGroup)view.getParent()).removeView(view);

}

super.onDestroyView();

}

onDestroy()

销毁fragment对象
跟activity类似了。


onDetach()

Fragment和Activity解除关联的时候调用。
脱离activity


Fragment实现方式

静态方法

以repair为例

动态方法

以新人项目为例

Fragment和Activity之间的区别

生命周期不同

得到context的方法不同 Fragment需要调用getActivity() Activity只需要直接Activity.this就可以了.

在使用butterknife 的时候 在Activity中通过 Retrofit.bind(this);

fragment通过Retrofit.bind(this,view);

Fragment与 Activity之间的通信

以repair为例

Fragment -> Activity

在Fragment中创建Interface

/**
* Fragment与Activity通信的接口
* 用于传递输入的电话号码
*/
public interface InputMobileListener {
//传入电话号码给Activity
void inputPhoneNumberNextAction(String mobile);
}


这里在Fragment中定义了一个接口,在接口中写了一个方法以保存手机号的字符串为参数,

在Fragment所依赖的Activity中实现接口并重写方法

public class ForgetPwdActivity implements InputMobileFragment.InputMobileListener{
······
@Override
public void inputPhoneNumberNextAction(final String mobile) {
//替换Fragment
replaceFragment(R.id.frame_layout, checkCodeFragment, null);
//请求后台为该号码发送验证码
sendVerifyCode(mobile);
}
}


在Activity中重写该方法,在方法体中接收到手机号并进行一系列操作,此处是发送验证码

4000

在Fragment中重写回调方法

Fragment与Activity一样,在创建和销毁时都会回调一系列回调方法,而onAttach方法是在Fragment创建之初回调的,所以我们重写onAttach方法便可以保证Fragment全程皆能调用接口中的方法。

@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
//抓取异常,在父Activity未实现接口时抛出。
try {
//Java的多态特性之一,父类引用子类对象
InputMobileListener inputMobileListener = (InputMobileListener) activity;
} catch (ClassCastException e) {
throw new ClassCastException(activity.toString() + getString(R.string.implementSPN));
}
}


在Fragment中通过接口引用Activity对象调用接口方法

public void onButtonClicked(){
//获取手机号输入框中的手机号
String mobile = mobileEdit.getText().toString().trim();
//此处应对获取的手机号做正则匹配,无关代码便不列出了
//调用重写后的接口方法
inputMobileListener = (InputMobileListener) activity;
}


Activity -> Fragment

public static class MainActivity extends Activity
implements HeadlinesFragment.OnHeadlineSelectedListener{
...

public void onArticleSelected(int position) {
// 用户选中HeadlinesFragment中的头标题后
// 做一些必要的业务操作

ArticleFragment articleFrag = (ArticleFragment)
getSupportFragmentManager().findFragmentById(R.id.article_fragment);

if (articleFrag != null) {
// 如果 article frag 不为空,那么我们在同时显示两个fragmnet的布局中...

// 调用ArticleFragment中的方法去更新它的内容
articleFrag.updateArticleView(position);
} else {
// 否则,我们就是在仅包含一个fragment的布局中并需要交换fragment...

// 创建fragment并给他一个跟选中的文章有关的参数
ArticleFragment newFragment = new ArticleFragment();
Bundle args = new Bundle();
args.putInt(ArticleFragment.ARG_POSITION, position);
newFragment.setArguments(args);

FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();

// 用这个fragment替换任何在fragment_container中的东西
// 并添加事务到back stack中以便用户可以回退到之前的状态
transaction.replace(R.id.fragment_container, newFragment);
transaction.addToBackStack(null);

// 提交事务
transaction.commit();
}
}
}


Fragment与Fragment之间的通信

就是通过Fragment1 -> Activity -> Fragment2的方式进行的

FragmentManager

manager的概念

用法

Activity需要继承FragmentActivity(直接继承 AppCompatActivity 也可以)

给PagerAdapter传入FragmentManager

WaterfallPagerAdapter pagerAdapter = new WaterfallPagerAdapter(getSupportFragmentManager());


viewPager 和 tabLayout

ViewPager

java



xml



FragmentPagerAdapter



数据域

FragmentList

titleList

方法

add()

getItem(int)重载

getCount()重载

getPageTitile()重载

tabLayout

java



XML

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  Android