当前位置: 首页 > news >正文

架构师成长之路-基于android fragment通信的面向对象的万能接口

前言

  开发一个app时,常用Activity和Fragment,由于操作方便Fragment越来越受欢迎,这样就避免不了Activity和Fragment、Fragment和Fragment之间的通信,我们知道Fragment和Fragment之间不能直接进行通信的,而是通过Activity进行通信。那么Activity和Fragment有几种通信方式呢,它们的优缺点又是什么?

一 常用的Activity和Fragment几种通信方式

1 通过Handle
在Activity创建一个Handle机制实例,然后在创建Fragment的通过构造方法把Handle实例传递过去,这样Fragment就可以往Activity传送数据了。但这样如下缺点:
(1)Activity和Fragment之间耦合性增大;
(2)Activity处理后的结果不能实时反馈给Fragment;
(3)提高了内存泄漏风险;
2 使用static变量
缺点很明确增大内存消耗;
3 使用广播
分别在Activity和Fragment中注册广播,这样就可以实现通信了。其缺点:
(1)性能差,有延迟,用户体验就会差;
(2)一般使用的是标准广播,一个发送,多个接收者,大材小用,性能差;
(3)代码冗余;
(4)传播的数据有限;
4 EventBus 、rxBus(俗称万能油)
其使用方法参考官方文档,其优点就是实用起来简单方便,其缺点也很明显:
(1)EventBus 、rxBus其内部使用的是反射机制,那么其性能就会降低;
(2)代码维护困难(不熟悉的项目代码的新人很难找到实现的方法在是如何调用的);
(3)数据返回困难,它们是单向传递;
5 普通接口
在Fragment写一个接口,让Activity去实现这个接口,通过这个接口把Activity与Fragment绑定在一起,这样Activity和Fragment实时进行通信了,其实谷歌推荐就是这么干的,由于每个Fragment都写一个接口,就会造成代码冗余;如果Fragment少的话还好,多的话,Activity实现多个接口,显得Activity头部很大,况且接口的命名也是一个问题;

二 万能接口

如果在5的基础能够解决代码冗余、接口命名就好了,我们知道一个函数包括函数名、函数体、参数、返回值,那么就可以通过搭建一个简单的框架实现上述问题。
1 建立基类
public abstract  class Function {
    /** 方法名 */
    public String mFunctionName ;

    public Function (String funName){
        this.mFunctionName = funName;
    }

}

2 建立无(有)参数无(有)四类

/**
 *  无参无返回值
 * <h3>Description</h3>
 * TODO
 * <h3>Author</h3> luzhenbang
 * <h3>Date</h3> 2018/1/4 16:13
 * <h3>Copyright</h3> Copyright (c)2018 Shenzhen TL  Co., Ltd. Inc. All rights reserved.
 */
public abstract class FunctionNoParamNoResult extends Function {

    public FunctionNoParamNoResult(String funName) {
        super(funName);
    }

    public abstract void function();

}

 

/**
 *  有参无返回值
 * <h3>Description</h3>
 * TODO
 * <h3>Author</h3> luzhenbang
 * <h3>Date</h3> 2018/1/4 16:13
 * <h3>Copyright</h3> Copyright (c)2018 Shenzhen TL  Co., Ltd. Inc. All rights reserved.
 */

public abstract class FunctionWithParamOnly<Param> extends Function {
    public FunctionWithParamOnly(String funName) {
        super(funName);
    }
    public abstract void function(Param param);
}    

 

/**
 *  有参有返回值
 * <h3>Description</h3>
 * TODO
 * <h3>Author</h3> luzhenbang
 * <h3>Date</h3> 2018/1/4 16:13
 * <h3>Copyright</h3> Copyright (c)2018 Shenzhen TL  Co., Ltd. Inc. All rights reserved.
 */

public abstract class FunctionWithParamWithResult<Result,Param> extends Function {
    public FunctionWithParamWithResult(String funName) {
        super(funName);
    }
    public abstract Result function(Param param);
}

 

/**
 *  无参有返回值
 * <h3>Description</h3>
 * TODO
 * <h3>Author</h3> luzhenbang
 * <h3>Date</h3> 2018/1/4 16:13
 * <h3>Copyright</h3> Copyright (c)2018 Shenzhen TL  Co., Ltd. Inc. All rights reserved.
 */

public abstract class FunctionWithResultOnly<Result> extends Function {
    public FunctionWithResultOnly(String funName) {
        super(funName);
    }
    public abstract Result function();
}

  

3 建立管理器类

public class FunctionManager {

private static FunctionManager instance = null;
    public static final String TAG = FunctionManager.class.getSimpleName() + "-------->";

    /**容器,用来存储方法名字 key 对应的方法名 value 对应的是 参数返回对象*/
    private HashMap<String,FunctionWithParamWithResult> mFunctionWithParamWithResultHashMap = null;
    private HashMap<String,FunctionWithParamOnly> mFunctionWithParamsOnlyHashMap = null;
    private HashMap<String,FunctionWithResultOnly> mFunctionWithResultOnlyHashMap = null;
    private HashMap<String,FunctionNoParamNoResult> mFunctionNoParamNoResultHashMap = null;


    private FunctionManager() {
mFunctionNoParamNoResultHashMap = new HashMap<>();
        mFunctionWithParamWithResultHashMap = new HashMap<>();
        mFunctionWithParamsOnlyHashMap = new HashMap<>();
        mFunctionWithResultOnlyHashMap = new HashMap<>();
    }


public static FunctionManager getInstance() {
if (null == instance){
instance = new FunctionManager();
        }
return instance;
    }


/**
     * 添加无参无返回值的方法
     * @return
     */
    public FunctionManager addFunction(FunctionNoParamNoResult function){
mFunctionNoParamNoResultHashMap.put(function.mFunctionName,function);
        return this;
    }


/**
     * 添加有返回值的方法
     * @return
     */
    public FunctionManager addFunction(FunctionWithResultOnly function){
mFunctionWithResultOnlyHashMap.put(function.mFunctionName,function);
        return this;
    }


/**
     * 添加有参数的方法
     * @return
     */
    public FunctionManager addFunction(FunctionWithParamOnly function){
mFunctionWithParamsOnlyHashMap.put(function.mFunctionName,function);
        return this;
    }



/**
     * 添加有参有返回值的方法
     * @return
     */
    public FunctionManager addFunction(FunctionWithParamWithResult function){
mFunctionWithParamWithResultHashMap.put(function.mFunctionName,function);
        return this;
    }


/**
     * 调用无返回值无参数的方法
     * @param funName
     */
    public void invokeNoAll (String funName) throws NullPointerException {
if (TextUtils.isEmpty(funName)){
            Log.e(TAG,"funName is null !");
        }else {
if ( null != mFunctionNoParamNoResultHashMap){
                FunctionNoParamNoResult function =  mFunctionNoParamNoResultHashMap.get(funName);
                if (null != function){
                    function.function();
                }else {
                    Log.e(TAG,"function is null !");
                }
            }else {
throw new NullPointerException("mFunctionNoParamNoResultHashMap can not be null ,please first init FunctionManager !");
            }
        }
    }


/**
     * 调用有参数的方法
     * @param funName
     */
   public <Param> void invokeWithParamOnly (String funName,Param param) throws NullPointerException {
if (TextUtils.isEmpty(funName)){
            Log.e(TAG,"funName is null !");
        }else {
if ( null != mFunctionWithParamsOnlyHashMap){
                FunctionWithParamOnly<Param> function =  mFunctionWithParamsOnlyHashMap.get(funName);
                if (null != function){
                    function.function(param);
                }else {
                    Log.e(TAG,"function is null !");
                }
            }else {
throw new NullPointerException("mFunctionWithParamsOnlyHashMap can not be null ,please first init FunctionManager !");
            }
        }
    }


/**
     * 调用有返回值的方法
     * @param funName
     */
   public <Result> Result invokeWithResultOnly (String funName, Class<Result> c) throws NullPointerException {
if (TextUtils.isEmpty(funName)){
            Log.e(TAG,"funName is null !");
        }else {
if ( null != mFunctionWithResultOnlyHashMap){
                FunctionWithResultOnly function =  mFunctionWithResultOnlyHashMap.get(funName);
                if (null != function){
if ( null != c){
return c.cast(function.function());
                    }else {
return (Result) function.function();
                    }
                }else {
                    Log.e(TAG,"function is null !");
                }
            }else {
throw new NullPointerException("mFunctionWithParamsOnlyHashMap can not be null ,please first init FunctionManager !");
            }
        }
return null;

   }

/**
     * 调用有参数有返回值的方法
     * @param funName
     */
    public <Result,Param> Result invokeWithAll (String funName, Class<Result> c,Param param) throws NullPointerException {
if (TextUtils.isEmpty(funName)){
            Log.e(TAG,"funName is null !");
        }else {
if ( null != mFunctionWithParamWithResultHashMap){
                FunctionWithParamWithResult<Result,Param> function =  mFunctionWithParamWithResultHashMap.get(funName);
                if (null != function){
if ( null != c){
return c.cast(function.function(param));
                    }else {
return function.function(param);
                    }
                }else {
                    Log.e(TAG,"function is null !");
                }
            }else {
throw new NullPointerException("mFunctionWithParamsOnlyHashMap can not be null ,please first init FunctionManager !");
            }
        }
return null;

    }
}

4 在Activity写一个方法

public void setFunctionForFragment(String tag){
if (TextUtils.isEmpty(tag)){
        Log.e(MainActivity.class.getSimpleName(),"tag is null !");
        return;
    }
    BaseFragment fragment = (BaseFragment) fm.findFragmentByTag(tag);
    FunctionManager functionManager = FunctionManager.getInstance();

    functionManager.addFunction(new FunctionNoParamNoResult(FunctionNoParamNoResult) {
@Override
        public void function() {
            Toast.makeText(MainActivity.this, "无参无返回值", Toast.LENGTH_SHORT).show();
        }
    });

    functionManager.addFunction(new FunctionWithResultOnly<String>(FunctionWithResultOnly) {
@Override
        public String function() {
return "无参有返回值";
        }
    });

    functionManager.addFunction(new FunctionWithParamOnly<String>(FunctionWithParamOnly) {
@Override
        public void function(String o) {
            Toast.makeText(MainActivity.this, o, Toast.LENGTH_SHORT).show();
        }
    });

    functionManager.addFunction(new FunctionWithParamWithResult<String,String>(FunctionWithParamWithResult) {
@Override
        public String function(String o) {
return o;
        }
    });

    fragment.setFunctionManager(functionManager);
}

5 编辑一个Fragment基类(实用Fragment的时候直接集成该类就可以了),并绑定上面建立的接口

public class BaseFragment extends Fragment{

public FunctionManager mFunctionManager;
    private MainActivity mainActivity;

    public void setFunctionManager(FunctionManager mFunctionManager) {
this.mFunctionManager = mFunctionManager;
    }

@Override
    public void onAttach(Context context) {
super.onAttach(context);
        if (context instanceof MainActivity) {
mainActivity  = (MainActivity) context;
            mainActivity.setFunctionForFragment(getTag());
        } else {
throw new RuntimeException(context.toString()
                    + " must implement OnFragmentInteractionListener");
        }
    }

@Override
    public void onDetach() {
super.onDetach();
        mainActivity = null;
    }

}

6 继承BaseFragment

public class NoParamNoResultFragment extends BaseFragment {

private Handler  mHandler;

    public NoParamNoResultFragment(Handler  handler) {
// Required empty public constructor
        this.mHandler  = handler;
    }


@Override
    public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
        if (getArguments() != null) {
        }
    }

@Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
// Inflate the layout for this fragment
        View  view  = inflater.inflate(R.layout.fragment_no_param_no_result, container, false);

        view.findViewById(R.id.txt_handle).setOnClickListener(new View.OnClickListener() {
@Override
            public void onClick(View v) {
                Message  message = mHandler.obtainMessage();
                message.what = 0x123;
                message.obj = "handle 通信";
                mHandler.sendMessage(message);
            }
        });


        view.findViewById(R.id.txt_noALl).setOnClickListener(new View.OnClickListener() {
@Override
            public void onClick(View v) {
mFunctionManager.invokeNoAll(MainActivity.FunctionNoParamNoResult);

            }
        });
        view.findViewById(R.id.txt_result).setOnClickListener(new View.OnClickListener() {
@Override
            public void onClick(View v) {
//                mFunctionManager.invokeNoAll(MainActivity.FunctionNoParamNoResult);
                String result =  mFunctionManager.invokeWithResultOnly(MainActivity.FunctionWithResultOnly,String.class);
                Toast.makeText(getActivity(), result, Toast.LENGTH_SHORT).show();
            }
        });
        view.findViewById(R.id.txt_param).setOnClickListener(new View.OnClickListener() {
@Override
            public void onClick(View v) {
mFunctionManager.invokeWithParamOnly(MainActivity.FunctionWithParamOnly,"有参无返回值");
            }
        });
        view.findViewById(R.id.txt_withAll).setOnClickListener(new View.OnClickListener() {
@Override
            public void onClick(View v) {
                String result =  mFunctionManager.invokeWithAll(MainActivity.FunctionWithParamWithResult,String.class,"有参有返回值");
                Toast.makeText(getActivity(), result, Toast.LENGTH_SHORT).show();
            }
        });
        return view;
    }
}

说明,上述在点击控件的时候,会触发在Activity添加的方法,这样就可以实现Activity与Fragment实时通信,近而实现Fragment之间的通信  

https://gitee.com/lzbgit/AppInterface  

 

转载于:https://www.cnblogs.com/ganchuanpu/p/9249764.html

相关文章:

  • python学习之老男孩python全栈第九期_day004知识点总结
  • 人人快速开发平台
  • shell执行和crontab执行结果不一样的问题
  • 面对谷歌AI,这真的不是拔电线就能够解决的
  • Linux中使用sendmail发送邮件,指定任意邮件发送人
  • linux基础之Vim
  • python如何解决MD5对文件加密出现粘包的情况
  • js获取url锚链,操作class和id的显示和隐藏
  • 【活动预告】Hi,游戏开发者,这有一份您的邀请函
  • Ocelot(v7.0.6)使用小记
  • 如何让机器向“时尚达人”学习?阿里做了个“实用”的图像数据集
  • 机器学习二分类模型评价指标:准确率\召回率\特异度等
  • Javascript装饰器的妙用
  • 前端框架三巨头:React仍是老大,Vue 增长率最高
  • 用laravel dingo/api创建产品api
  • Angularjs之国际化
  • css属性的继承、初识值、计算值、当前值、应用值
  • Debian下无root权限使用Python访问Oracle
  • django开发-定时任务的使用
  • eclipse的离线汉化
  • HashMap剖析之内部结构
  • iOS动画编程-View动画[ 1 ] 基础View动画
  • Javascripit类型转换比较那点事儿,双等号(==)
  • java正则表式的使用
  • overflow: hidden IE7无效
  • Python语法速览与机器学习开发环境搭建
  • ReactNative开发常用的三方模块
  • Vue.js-Day01
  • vue+element后台管理系统,从后端获取路由表,并正常渲染
  • 基于Javascript, Springboot的管理系统报表查询页面代码设计
  • 计算机常识 - 收藏集 - 掘金
  • 检测对象或数组
  • 协程
  • 一起来学SpringBoot | 第三篇:SpringBoot日志配置
  • Semaphore
  • 阿里云重庆大学大数据训练营落地分享
  • 好程序员web前端教程分享CSS不同元素margin的计算 ...
  • ​​​​​​​sokit v1.3抓手机应用socket数据包: Socket是传输控制层协议,WebSocket是应用层协议。
  • ​520就是要宠粉,你的心头书我买单
  • ​软考-高级-系统架构设计师教程(清华第2版)【第20章 系统架构设计师论文写作要点(P717~728)-思维导图】​
  • (0)Nginx 功能特性
  • (1)(1.9) MSP (version 4.2)
  • (9)目标检测_SSD的原理
  • (附源码)springboot美食分享系统 毕业设计 612231
  • (附源码)ssm教师工作量核算统计系统 毕业设计 162307
  • (四)鸿鹄云架构一服务注册中心
  • (原)本想说脏话,奈何已放下
  • (原創) 如何安裝Linux版本的Quartus II? (SOC) (Quartus II) (Linux) (RedHat) (VirtualBox)
  • ***检测工具之RKHunter AIDE
  • ./configure,make,make install的作用
  • .NET Framework .NET Core与 .NET 的区别
  • .net/c# memcached 获取所有缓存键(keys)
  • .NET6使用MiniExcel根据数据源横向导出头部标题及数据
  • .net企业级架构实战之7——Spring.net整合Asp.net mvc
  • .w文件怎么转成html文件,使用pandoc进行Word与Markdown文件转化