手机平板要兼顾—-探究碎片
碎片是什么
碎片(Fragment)是一种可以嵌入在活动当中的UI片段,它能让程序更加合理和充分地利用大屏幕的空间,因而在平板上应用的非常广泛
系统内置的android.app.Fragment,一个是support-v4库中的Fragment,因为它可以让碎片在所有Android系统版本中保持功能一致性。
动态添加碎片主要分为五步
- 创建待添加的碎片实例
- 获取FragmentManager,在活动中可以直接通过调用getSupportFragmentManager()方法得到
- 开启一个事务,通过调用beginTransaction()方法开启
- 想容器内添加或替换碎片,一般使用replace()方法实现,需要传入容器的id和待添加的碎片实例
提交事务,调用 commit()方法来完成
private void replaceFragment(Fragment fragment) {
FragmentManager fragmentManager = getSupportFragmentManager();
FragmentTransaction transaction = fragmentManager.beginTransaction();
transaction.replace(R.id.right_fragment , fragment);
transaction.addToBackStack(null);//接收名字用于描述返回栈的状态
transaction.commit();
}
碎片和活动之间进行通信
RightFragment rightFragment = (RightFragment) getFragmentManager().findFragmentById(R.id.right_fragment);
调用FragmentManager的findFragmentById()方法,可以在活动中得到相应碎片的实例,然后就能轻松地调用碎片里的方法
在每个碎片中都可以通过调用getActivity()方法来得到和当前碎片相关联的活动实例
MainActivity activity = (MainActivity) getActivity();
碎片和碎片之间的通信
在一个碎片中可以得到与它相关联的活动,然后通过这个活动去获取另外一个碎片的实例,这样也就实现了不同碎片之间的通信功能
碎片的状态和回调
- 运行状态:当一个碎片是可见的,并且它所关联的活动正处于运行状态时,该碎片也处于运行状态
- 暂停状态:当一个活动进入暂停状态时(由于另一个未占满屏幕的活动被添加到了栈顶),与它相关联的可见碎片就会进入到暂停状态
- 停止状态:当一个活动进入停止状态时,与它相关联的碎片就会进入到停止状态,或者通过调用FragmentTransaction的 remove()、replace()方法将碎片从活动中移除,但如果在事务提交之前调用addToBackStack()方法,这时的碎片也会进入到停止状态。总的来说,进入停止状态的碎片对用户来说是完全不可见的,有可能会被系统回收
- 销毁状态:碎片总是依附活动而存在的,因此当活动被销毁时,与它相关联的碎片就会进入到销毁状态。或者通过调用FragmentTransaction的 remove()、replace()方法将碎片从活动中移除,但如果在事务提交之前没有调用addToBackStack()方法,这时的碎片也会进入到销毁状态
回调方法:
- onAttach() 当碎片和活动创建关联的时候调用
- onCreateView() 为碎片创建视图(加载布局)时调用
- onActivityCreated() 确保与碎片相关联的活动一定已经创建完毕的时候调用
- onDestroyView() 当与碎片相关联的视图被移除的时候调用
- onDetach() 当碎片和活动解除关联的时候调用
fragment生命周期
fragment与Activity的生命周期对比
fragment的优点
Fragment可以使你能够将activity分离成多个可重用的组件,每个都有它自己的生命周期和UI。
Fragment可以轻松得创建动态灵活的UI设计,可以适应于不同的屏幕尺寸。从手机到平板电脑。
Fragment是一个独立的模块,紧紧地与activity绑定在一起。可以运行中动态地移除、加入、交换等。
Fragment提供一个新的方式让你在不同的安卓设备上统一你的UI。
Fragment 解决Activity间的切换不流畅,轻量切换。
Fragment 替代TabActivity做导航,性能更好。
Fragment 在4.2.版本中新增嵌套fragment使用方法,能够生成更好的界面效果。
Fragment做局部内容更新更方便,原来为了到达这一点要把多个布局放到一个activity里面,现在可以用多Fragment来代替,只有在需要的时候才加载Fragment,提高性能
步骤
public class RightFragment extends Fragment {
public static final String TAG = "RightFragment";
@Override
public void onAttach(Context context) {
super.onAttach(context);
Log.d(TAG, "onAttach: ");
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d(TAG, "onCreate: ");
}
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.right_fragment , container , false);
return view;
}
@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
Log.d(TAG, "onActivityCreated: ");
}
@Override
public void onStart() {
super.onStart();
Log.d(TAG, "onStart: ");
}
@Override
public void onResume() {
super.onResume();
Log.d(TAG, "onResume: ");
}
@Override
public void onPause() {
super.onPause();
Log.d(TAG, "onPause: ");
}
@Override
public void onStop() {
super.onStop();
Log.d(TAG, "onStop: ");
}
@Override
public void onDestroyView() {
super.onDestroyView();
Log.d(TAG, "onDestroyView: ");
}
@Override
public void onDestroy() {
super.onDestroy();
Log.d(TAG, "onDestroy: ");
}
@Override
public void onDetach() {
super.onDetach();
Log.d(TAG, "onDetach: ");
}
}
在RightFragment中的每一个回调方法里都加入了打印日志的代码:
01-03 04:04:14.261 3498-3498/com.demo.fragmenttest D/RightFragment: onAttach:
01-03 04:04:14.261 3498-3498/com.demo.fragmenttest D/RightFragment: onCreate:
01-03 04:04:14.262 3498-3498/com.demo.fragmenttest D/RightFragment: onCreateView:
01-03 04:04:14.262 3498-3498/com.demo.fragmenttest D/RightFragment: onActivityCreated:
01-03 04:04:14.262 3498-3498/com.demo.fragmenttest D/RightFragment: onStart:
01-03 04:04:14.266 3498-3498/com.demo.fragmenttest D/RightFragment: onResume:
当RightFragment第一次被加载到屏幕上时,会一次执行 onAttach()、onCreate()、onCreateView()、onActivityCreated()、onStart()、onResume()方法。然后点击LeftFragment中的按钮,此时打印信息如下
01-03 04:07:07.001 3498-3498/com.demo.fragmenttest D/RightFragment: onPause:
01-03 04:07:07.002 3498-3498/com.demo.fragmenttest D/RightFragment: onStop:
01-03 04:07:07.002 3498-3498/com.demo.fragmenttest D/RightFragment: onDestroyView:
此时RightFragment进入了停止状态,因此onPause()、onStop()、onDestroyView()方法会得到执行。如果在替换的时候没有调用addToBackStack()方法,此时的RightFragment就会进入销毁状态,onDestroy()和onDetach()方法就会得到执行。接着按下back键,RightFragment会重新回到屏幕,打印信息如下:
01-03 04:09:54.047 3498-3498/com.demo.fragmenttest D/RightFragment: onActivityCreated:
01-03 04:09:54.047 3498-3498/com.demo.fragmenttest D/RightFragment: onStart:
01-03 04:09:54.047 3498-3498/com.demo.fragmenttest D/RightFragment: onResume:
由于RightFragment重新回到了运行状态,因此onActivityCreated()、onStart()、onResume()方法会得到执行。注意此时onCreate()、和onCreateView()方法并不会执行,因为我们借助了addToBackStack()方法使得RightFragment和它的视图并没有被销毁
再次按下Back键退出程序,打印信息如下:
01-03 04:10:43.372 3498-3498/com.demo.fragmenttest D/RightFragment: onPause:
01-03 04:10:43.373 3498-3498/com.demo.fragmenttest D/RightFragment: onStop:
01-03 04:10:43.373 3498-3498/com.demo.fragmenttest D/RightFragment: onDestroyView:
01-03 04:10:43.374 3498-3498/com.demo.fragmenttest D/RightFragment: onDestroy:
01-03 04:10:43.374 3498-3498/com.demo.fragmenttest D/RightFragment: onDetach:
执行onPause、onStop、onDestroyView、onDestroy、onDetach,最终将活动和碎片一起销毁