Fragment初学8——Fragment在Android开发中的应用2

简介: Fragment都是依附于Activity的,通信方式大致也分为如下几种:如果Activity中包含自己管理的Fragment的引用,可以通过直接引用访问所有的Fragment的public方法 如果Activity中未保存任何Fragment的引用,那么可以通过 getFragmentManager.findFragmentByTag()或者findFragmentById()获得任何Fragment实 例,然后进行操作。

Fragment都是依附于Activity的,通信方式大致也分为如下几种:

  1. 如果Activity中包含自己管理的Fragment的引用,可以通过直接引用访问所有的Fragment的public方法
  2.  如果Activity中未保存任何Fragment的引用,那么可以通过 getFragmentManager.findFragmentByTag()或者findFragmentById()获得任何Fragment实 例,然后进行操作。
  3. 在Fragment中可以通过getActivity得到当前绑定的Activity的实例,然后进行操作。

注意:如果在Fragment中需要Context,可以通过调用getActivity(),如果该Context需要在Activity被销毁后还存在,则使用getActivity().getApplicationContext()。

因为要考虑Fragment的重复使用,所以必须降低Fragment与Activity的耦合,而且Fragment更不应该直接操作别的Fragment,毕竟Fragment操作应该由它的管理者Activity来决定。

下面通过两种方式分别重构FragmentOne和FragmentTwo的点击事件,以及Activity对点击事件的响应:


FragmentOne.java如下

public class FragmentOne extends Fragment {
    private Button mButton;

    /**
     * 设置按钮点击的回调接口
     * 
     */
    public interface BtnOneClickListener {
        void onBtnOneClick();
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        // return inflater.inflate(R.layout.fragment_one, container, false);
        View view = inflater.inflate(R.layout.fragment_one, container, false);
        mButton = (Button) view.findViewById(R.id.btn_fragment_one);
        mButton.setOnClickListener(new OnClickListener() {
            // 由所属的Activity处理
            @Override
            public void onClick(View v) {
                if (getActivity() instanceof BtnOneClickListener) {
                    ((BtnOneClickListener) getActivity()).onBtnOneClick();
                }
            }
        });
        return view;
    }
}
现在FragmentOne不和任何Activity耦合,任何Activity都可以使用;同时声明了一个接口回调其点击事件,想要管理其点击事件的Activity实现此接口就即可。可以看到我们在onClick中首先判断了当前绑定的Activity是否实现了该接口,如果实现了则调用。

FragmentTwo.java类如下
public class FragmentTwo extends Fragment {

    private Button mButton;
    private BtnTwoClickListener mBtnTwoClickListener;

    public interface BtnTwoClickListener {
        void onBtnTwoClick();
    }
      //设置回调接口  
    public void setBtnTwoClickListener(BtnTwoClickListener btnTwoClickListener)  
    {  
        this.mBtnTwoClickListener = btnTwoClickListener;  
    } 
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        // return inflater.inflate(R.layout.fragment_one, container, false);
        View view = inflater.inflate(R.layout.fragment_two, container, false);
        mButton = (Button) view.findViewById(R.id.btn_fragment_two);
        mButton.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                 if(mBtnTwoClickListener != null)  
                    {  
                     mBtnTwoClickListener.onBtnTwoClick();  
                    }  
            }
        });
        return view;
    }
}


代码大致和FragmentOne结构相同,与FragmentOne不同的是我们提供了setListener这样的方法,意味着Activity不仅需要实现该接口,还必须显示调用mButton.setBtnTwoClickListener(this)。

MainActivity类如下
public class MainActivity extends Activity implements BtnOneClickListener,BtnTwoClickListener{

    private FragmentOne mFragmentOne;  
    private FragmentTwo mFragmentTwo;  
    private FragmentThree mFragmentThree; 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        // getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
        // WindowManager.LayoutParams.FLAG_FULLSCREEN);
        setContentView(R.layout.activity_main);
        mFragmentOne=new FragmentOne();
        FragmentManager fragmentManager = getFragmentManager();
        FragmentTransaction transaction = fragmentManager.beginTransaction();
        transaction.add(R.id.framelayout_fragment_main, mFragmentOne, "ONE");
        transaction.commit();
    }

    @Override
    public void onBtnOneClick() {
        if (mFragmentTwo == null)  
        {  
            mFragmentTwo = new FragmentTwo();  
            mFragmentTwo.setBtnTwoClickListener(this);  
        }  
        FragmentManager fragmentManager = getFragmentManager();  
        FragmentTransaction transaction = fragmentManager.beginTransaction();  
        transaction.replace(R.id.framelayout_fragment_main, mFragmentTwo, "TWO");  
        transaction.addToBackStack(null);  
        transaction.commit(); 
    }
    @Override
    public void onBtnTwoClick() {
        if (mFragmentThree == null)  
        {  
            mFragmentThree = new FragmentThree();  

        }  
        FragmentManager fm = getFragmentManager();  
        FragmentTransaction transaction = fm.beginTransaction();  
        transaction.hide(mFragmentTwo);  
        transaction.add(R.id.framelayout_fragment_main, mFragmentThree, "THREE");  
        transaction.addToBackStack(null);  
        transaction.commit(); 
    }
}
通过重构,项目效果和上一节的效果是一样的,这两种通信方式都是值得推荐的,我建议还是选择第二种。虽然Fragment和Activity可以通过getActivity与 findFragmentByTag或者findFragmentById进行任何操作,甚至在Fragment里面操作另外的Fragment,但是除非万不得已还是别用。Activity担任的是Fragment间类似总线一样的角色,应当由它决定Fragment的操作。另外Fragment不能响应Intent,但是Activity可以,Activity可以接收Intent,然后根据参数判断显示哪个 Fragment。
可说了这么多,我们有没有发现,这些都是理想的情况下,一旦运行时配置发生变化,例如屏幕发生旋转,屏幕会重新加载,很多人觉得强制设置屏幕方向不变就可以了,以前我也这样做,但是当应用被置于后台(例如用户点击了home键)长时间没有返回的时候,应用也会被重新启动。 比如上例:如果把上面的例子置于FragmentThree界面,然后处于后台状态,长时间后你会发现当你再次通过home打开时,上面 FragmentThree与FragmentOne叠加在一起,这就是因为你的Activity重新启动,在原来的FragmentThree上又绘制 了一个FragmentOne。

为了体现一下效果,再写个简单的FragmentOne.java

public class FragmentOne extends Fragment {

     private static final String TAG = "xmr";  


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        // return inflater.inflate(R.layout.fragment_one, container, false);
        View view = inflater.inflate(R.layout.fragment_one, container, false);

        return view;
    }
    @Override  
    public void onCreate(Bundle savedInstanceState)  
    {  
        // TODO Auto-generated method stub  
        super.onCreate(savedInstanceState);  

        Log.i(TAG, "onCreate");  
    }  

    @Override  
    public void onDestroyView()  
    {  
        // TODO Auto-generated method stub  
        super.onDestroyView();  
        Log.i(TAG, "onDestroyView");  
    }  

    @Override  
    public void onDestroy()  
    {  
        // TODO Auto-generated method stub  
        super.onDestroy();  
        Log.i(TAG, "onDestroy");  
    }  

}

MainActivity.java
public class MainActivity extends Activity {

    private FragmentOne mFragmentOne;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        // getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
        // WindowManager.LayoutParams.FLAG_FULLSCREEN);
        setContentView(R.layout.activity_main);
        mFragmentOne = new FragmentOne();
        FragmentManager fragmentManager = getFragmentManager();
        FragmentTransaction transaction = fragmentManager.beginTransaction();
        transaction.add(R.id.framelayout_fragment_main, mFragmentOne, "ONE");
        transaction.commit();
    }

}
不断的旋转屏幕,你会发现每旋转一次屏幕,屏幕上就多了一个FragmentOne的实例,并且后台log会打印出许多套生命周期的回调。


其实在上一节 Fragment状态管理 已经提到过,当屏幕发生旋转时,Activity会重新启动,默认的Activity中的Fragment也会跟着Activity重新创建,这就造成当旋转的时候,本身存在的Fragment会重新启动,然后当执行Activity的onCreate时,又会再次实例化一个新的Fragment, 这就是出现的原因。

解决办法就是通过检查onCreate的参数Bundle savedInstanceState判断当前是否发生Activity的重新创建。

简单改一下代码,只有在savedInstanceState==null时,才进行创建Fragment实例:

public class MainActivity extends Activity {

    private FragmentOne mFragmentOne;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        // getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
        // WindowManager.LayoutParams.FLAG_FULLSCREEN);
        setContentView(R.layout.activity_main);
        if (savedInstanceState != null) {
            mFragmentOne = new FragmentOne();
            FragmentManager fragmentManager = getFragmentManager();
            FragmentTransaction transaction = fragmentManager
                    .beginTransaction();
            transaction
                    .add(R.id.framelayout_fragment_main, mFragmentOne, "ONE");
            transaction.commit();
        }
    }

}


现在无论进行多次旋转都只会有一个Fragment实例在Activity中。但是这并解决所有问题,例如当重新绘制时,Fragment发生重建,原本的数据如何保持?

其实和Activity类似,Fragment也有onSaveInstanceState的方法,在此方法中进行保存数据,然后在onCreate或者onCreateView或者onActivityCreated进行恢复都可以。

Fragment与ActionBar和MenuItem集成

Fragment可以添加自己的MenuItem到Activity的ActionBar或者可选菜单中。使用方法也很简单:

1、在Fragment的onCreate中调用setHasOptionsMenu(true);

2、然后在Fragment子类中实现onCreateOptionsMenu

3、如果希望在Fragment中处理MenuItem的点击,也可以实现onOptionsItemSelected,当然了Activity也可以直接处理该MenuItem的点击事件。


public class FragmentOne extends Fragment {

    private static final String TAG = "xmr";

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        // return inflater.inflate(R.layout.fragment_one, container, false);
        View view = inflater.inflate(R.layout.fragment_one, container, false);
        Button mButton = (Button) view.findViewById(R.id.btn_fragment_one);
        return view;
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        super.onCreate(savedInstanceState);
        setHasOptionsMenu(true);
    }

    @Override
    public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
        // TODO Auto-generated method stub
        super.onCreateOptionsMenu(menu, inflater);
        inflater.inflate(R.menu.menu_fragment, menu);
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()) {
        case R.id.item1_menu_fragment:
            Toast.makeText(getActivity(), "fragment1", Toast.LENGTH_SHORT)
                    .show();
            return true;
        case R.id.item2_menu_fragment:
            Toast.makeText(getActivity(), "fragment1", Toast.LENGTH_SHORT)
                    .show();
            return true;
        default:
            return true;
        }
    }

    @Override
    public void onDestroyView() {
        // TODO Auto-generated method stub
        super.onDestroyView();
        Log.i(TAG, "--onDestroyView");
    }

    @Override
    public void onDestroy() {
        // TODO Auto-generated method stub
        super.onDestroy();
        Log.i(TAG, "--onDestroy");
    }

}


public class MainActivity extends Activity {

    private FragmentOne mFragmentOne;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        // getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
        // WindowManager.LayoutParams.FLAG_FULLSCREEN);
        setContentView(R.layout.activity_main);
        if (savedInstanceState == null) {
            mFragmentOne = new FragmentOne();
            FragmentManager fragmentManager = getFragmentManager();
            FragmentTransaction transaction = fragmentManager
                    .beginTransaction();
            transaction
                    .add(R.id.framelayout_fragment_main, mFragmentOne, "ONE");
            transaction.commit();
        }
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // TODO Auto-generated method stub
         super.onCreateOptionsMenu(menu);
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }
      @Override  
        public boolean onOptionsItemSelected(MenuItem item)  
        {  
            switch (item.getItemId())  
            {  
            case R.id.action_settings:  
                Toast.makeText(this, "setting", Toast.LENGTH_SHORT).show();  
                return true;  
            default:  
                //如果希望Fragment自己处理MenuItem点击事件,一定不要忘了调用super.xxx  
                return super.onOptionsItemSelected(item);  
            }  
        } 
}


源代码


参考:


相关文章
|
4天前
|
Dart 前端开发 Android开发
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
|
1月前
|
搜索推荐 前端开发 API
探索安卓开发中的自定义视图:打造个性化用户界面
在安卓应用开发的广阔天地中,自定义视图是一块神奇的画布,让开发者能够突破标准控件的限制,绘制出独一无二的用户界面。本文将带你走进自定义视图的世界,从基础概念到实战技巧,逐步揭示如何在安卓平台上创建和运用自定义视图来提升用户体验。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开新的视野,让你的应用在众多同质化产品中脱颖而出。
67 19
|
1月前
|
JSON Java API
探索安卓开发:打造你的首个天气应用
在这篇技术指南中,我们将一起潜入安卓开发的海洋,学习如何从零开始构建一个简单的天气应用。通过这个实践项目,你将掌握安卓开发的核心概念、界面设计、网络编程以及数据解析等技能。无论你是初学者还是有一定基础的开发者,这篇文章都将为你提供一个清晰的路线图和实用的代码示例,帮助你在安卓开发的道路上迈出坚实的一步。让我们一起开始这段旅程,打造属于你自己的第一个安卓应用吧!
71 14
|
1月前
|
Java Linux 数据库
探索安卓开发:打造你的第一款应用
在数字时代的浪潮中,每个人都有机会成为创意的实现者。本文将带你走进安卓开发的奇妙世界,通过浅显易懂的语言和实际代码示例,引导你从零开始构建自己的第一款安卓应用。无论你是编程新手还是希望拓展技术的开发者,这篇文章都将为你打开一扇门,让你的创意和技术一起飞扬。
|
1月前
|
XML 存储 Java
探索安卓开发之旅:从新手到专家
在数字时代,掌握安卓应用开发技能是进入IT行业的关键。本文将引导读者从零基础开始,逐步深入安卓开发的世界,通过实际案例和代码示例,展示如何构建自己的第一个安卓应用。我们将探讨基本概念、开发工具设置、用户界面设计、数据处理以及发布应用的全过程。无论你是编程新手还是有一定基础的开发者,这篇文章都将为你提供宝贵的知识和技能,帮助你在安卓开发的道路上迈出坚实的步伐。
40 5
|
1月前
|
开发框架 Android开发 iOS开发
安卓与iOS开发中的跨平台策略:一次编码,多平台部署
在移动应用开发的广阔天地中,安卓和iOS两大阵营各占一方。随着技术的发展,跨平台开发框架应运而生,它们承诺着“一次编码,到处运行”的便捷。本文将深入探讨跨平台开发的现状、挑战以及未来趋势,同时通过代码示例揭示跨平台工具的实际运用。
148 3
|
1月前
|
XML 搜索推荐 前端开发
安卓开发中的自定义视图:打造个性化UI组件
在安卓应用开发中,自定义视图是一种强大的工具,它允许开发者创造独一无二的用户界面元素,从而提升应用的外观和用户体验。本文将通过一个简单的自定义视图示例,引导你了解如何在安卓项目中实现自定义组件,并探讨其背后的技术原理。我们将从基础的View类讲起,逐步深入到绘图、事件处理以及性能优化等方面。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧。
|
1月前
|
搜索推荐 前端开发 测试技术
打造个性化安卓应用:从设计到开发的全面指南
在这个数字时代,拥有一个定制的移动应用不仅是一种趋势,更是个人或企业品牌的重要延伸。本文将引导你通过一系列简单易懂的步骤,从构思你的应用理念开始,直至实现一个功能齐全的安卓应用。无论你是编程新手还是希望拓展技能的开发者,这篇文章都将为你提供必要的工具和知识,帮助你将创意转化为现实。
|
1月前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
2月前
|
搜索推荐 前端开发 Android开发
安卓开发中的自定义视图——打造个性化用户界面
在安卓应用开发的广阔天地里,自定义视图是实现个性化界面设计的重要手段。通过深入理解安卓绘图基础、触摸事件处理和布局机制,开发者可以突破标准控件的限制,创造出独一无二的用户体验。本文将引导你探索自定义视图的核心概念,提供实用的代码示例,并分享如何有效地解决开发过程中可能遇到的挑战。无论你是初学者还是有经验的开发者,这篇文章都将为你打开一扇通往创新和个性化的大门。