之所以把模板模式和桥接模式联系到一块儿说,是因为我最近写一个HBaseTemplate的时候突然发现按照模板模式去写,使用的时候必须继承自该类,很重量级,不优雅。于是对他进行改造,改造之后发现这不就是桥接模式吗?
先说一下什么是模板模式?
引用《设计模式之禅》中的模板模式的定义:定义一个操作中算法的骨架,而将一些步骤延迟到子类中,使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。
我个人通俗的解释是某件事有成型或者固有的操作流程,先干A,再干B,最后干C,而A,C是必须要干的,而且每个流程都一样,那么我们可以定义一个抽象类,把B这件事定义为一个抽象方法,让子类去实现,不同的B就有不同的子类,但是主流程没有变,而且也不用管A和C,只关注B就行了。
比如我们用HBase的时候,要先设置configuration、然后再获取connection,然后根据tablename获取HTable,根据HTable去做put,get操作,用完后关闭table。
public static HConnection CONNECTION = null; HTable table = null; try{ CONF = HBaseConfiguration.create(); CONNECTION = HConnectionManager.createConnection(CONF); table = (HTable) CONNECTION.getTable(tableName); Put rowPut = new Put(Bytes.toBytes(rowKey)); table.put(rowPut); }catch(Exception e){ if(table != null){ table.close(); } }
如以上步骤,只有7、 8两行是实际业务有用的操作,其他部分所有的都得写一遍,太费事了。
于是定义了一个抽象模板类
public abstract class HBaseTemplate { public void execute(String tableName) { Configuration CONF = null; HConnection CONNECTION = null; HTable table = null; try { CONF = HBaseConfiguration.create(); CONNECTION = HConnectionManager.createConnection(CONF); table = (HTable) CONNECTION.getTable(tableName); doInTable(table); } catch (Exception e) { if (table != null) { try { table.close(); } catch (Exception e1) { e1.printStackTrace(); } } } } public abstract void doInTable(HTable hTable); }
以上红色代码是实际要做的事情,实际用的时候继承该类,实现doInTable()方法就可以了,获取connection、关闭连接等都不用管了。
这就是一个典型的模板模式,类图如下:
什么是桥接模式?
以上模板模式存在的问题是每次使用都需要继承父类,很重量级。针对以上结构做了如下改造
public class HBaseTemplate { public void execute(String tableName,TableCallback tableCallback) { Configuration CONF = null; HConnection CONNECTION = null; HTable table = null; try { CONF = HBaseConfiguration.create(); CONNECTION = HConnectionManager.createConnection(CONF); table = (HTable) CONNECTION.getTable(tableName); tableCallback.doInTable(table); } catch (Exception e) { if (table != null) { try { table.close(); } catch (Exception e1) { e1.printStackTrace(); } } } } } public static interface TableCallback { void doInTable(HTableInterface table) throws Exception; } }
把之前的抽象方法变成了接口,这样用的时候可以这样:
HBaseTemplate hbaseTemplate = new HBaseTemplate(); hbaseTemplate.execute("tableName1", new TableCallback() { @Override public void doInTable(HTableInterface table) throws Exception { //do something... } });
也可以把匿名内部类的实现单独定义为一个类TableCallBackImpl,动态给hbaseTemplate注入进来,这样既灵活又不用继承。
以上是对模板模式的一个改造,以上只是针对获取htable后的操作的变化,但是如果获取htable的方法也变化怎么办,比如以后我要维护一个HTable的池,不每次都"获取table->用table->关闭连接",而是同一个tableName维护一定数量的table,每次从池里取,可以快速,并节省资源。可以再改造
public abstract class HBaseTemplate { public void execute(String tableName,TableCallback tableCallback) { Configuration CONF = null; HConnection CONNECTION = null; HTable table = null; try { getTable(tableName); tableCallback.doInTable(table); } catch (Exception e) { if (table != null) { try { table.close(); } catch (Exception e1) { e1.printStackTrace(); } } } } public static interface TableCallback { void doInTable(HTableInterface table) throws Exception; } public abstract HTable getTable(String tableName); }
这样就可以有HBasePoolTemplate、HBaseNotPoolTemplate实现了获取table维度的扩展。
其实也大可以把getTable()抽象方法改成接口类,动态注入进来。
写着写着感觉我们平常写Controller、Service类的时候就是在注入各种service、dao,头大了吧,其实设计模式只是一种理念、一种思想,体现到类图上可能会很像,甚至一样。理解这种思想,在实际工作中有意识的去应用,而不是随便乱堆代码就可以了。时间长了自然能写出很优雅的代码,设计模式神马的已经融入血液了,无意识的就知道什么场景该怎么写,已经不管他什么模式了。就像张三丰教张无忌太极剑,最后张无忌一招都不记得了,但却很厉害!
如果不是这么做还是按照之前模板模式的方案,那么每有一个htable的操作都要有一个子类HBaseDao1,HBaseDao2,如果再加上连接池,那么又要继承自HBaseDao1,HBaseDao2,就出现了HBasePoolDao1,HBasePoolDao2,如果有三个维度的变化那类的扩张速度就更多了。
桥接模式定义:将抽象和实现解耦,使得两者可以独立的变化。
桥接模式类图:
1.Bridge模式使用“对象间的组合关系”解耦了抽象和实现之间固有的绑定关系,使得抽象和实现可以沿着各自的维度来变化。
2.所谓抽象和实现沿着各自维度的变化,即“子类化”它们,得到各个子类之后,便可以任意组合它们。
3.Bridge模式的应用一般在“两个非常强的变化维度(以上例子的两个维度是:获取htable、应用htable做增删改)”,有时候即使有两个变化的维度,但是某个方向的变化维度并不剧烈——换言之两个变化不会导致纵横交错的结果,并不一定要使用Bridge模式。
相关推荐
1、文件内容:abrt-devel-2.1.11-60.el7.centos.rpm以及相关依赖 2、文件形式:tar.gz压缩包 3、安装指令: #Step1、解压 tar -zxvf /mnt/data/output/abrt-devel-2.1.11-60.el7.centos.tar.gz #Step2、进入解压后的目录,执行安装 sudo rpm -ivh *.rpm 4、安装指导:私信博主,全程指导安装
房地产公司绩效管理办法(精品)
网卡驱动-r8169-linux-官方驱动,能翻墙的也可以去外网下,这里只是赚个积分,本人有时候也需要下载别人的资源,望理解,外网地址:https://www.realtek.com/Download/ToDownload?type=direct&downloadid=3378
基于非线性干扰观测器的自适应滑模反演控制策略在机械臂模型中的应用:神经网络MATLAB仿真研究,(文献+程序)基于非线性干扰观测器的自适应滑模反演控制 机械臂模型 神经网络 matlab仿真 滑膜 带原班文献 ,关键词:非线性干扰观测器; 自适应滑模反演控制; 机械臂模型; 神经网络; MATLAB仿真; 滑膜控制; 原班文献,基于非线性干扰观测器的机械臂自适应滑模反演控制:matlab神经网络仿真及原班文献解读
"电力电子方向入门学习:单相PWM整流无桥图腾柱pfc技术Simulink仿真实践,电压调控及优化性能探索",单相PWM整流无桥图腾柱pfc,simulink仿真 输入电压220v有效值 输出电压500v纹波在1%以内 功率因数为1 电流THD<5% 开关频率20k 可作为电力电子方向入门学习~~ ,关键词:单相PWM整流;无桥图腾柱PFC;Simulink仿真;输入电压220V;输出电压500V;纹波;功率因数1;电流THD<5%;开关频率20k;电力电子方向入门学习。,"单相PWM整流桥技术:无桥图腾柱PFC的Simulink仿真入门学习"
员工晋升申请表
项目已获导师指导并通过的高分毕业设计项目,可作为课程设计和期末大作业,下载即用无需修改,项目完整确保可以运行。 包含:项目源码、数据库脚本、软件工具等,该项目可以作为毕设、课程设计使用,前后端代码都在里面。 该系统功能完善、界面美观、操作简单、功能齐全、管理便捷,具有很高的实际应用价值。 项目都经过严格调试,确保可以运行!可以放心下载 技术组成 语言:java 开发环境:idea 数据库:MySql8.0 部署环境:Tomcat(建议用 7.x 或者 8.x 版本),maven 数据库工具:navicat
FPGA实现CameraLink相机Base模式解码与HDMI高清视频输出方案,FPGA采集CameraLink相机Base模式 本文详细描述了FPGA采集CameraLink相机Base模式解码输出的实现设计方案,思路是这样的,CameraLink相机输入到FPGA板子,FPGA使用内部逻辑资源实现LVDS视频解码,解析出像素时钟、行同步信号、场同步信号、数据有效信号、以及像素数据,然后将视频转为Xilinx的AXI4-Sream的视频流,经VDMA送入DDR3缓存,然后读取出视频再经过AXI4-Sream to Video Out通过HDMI接口输出视频,这是Xilinx图像处理常用的套路,可谓相当精巧的方案 ,核心关键词:FPGA; CameraLink相机; Base模式; LVDS视频解码; 像素时钟; 同步信号; 像素数据; Xilinx AXI4-Sream; VDMA; DDR3缓存; HDMI接口视频输出。,"FPGA实现CameraLink相机Base模式视频解码与输出设计"
各岗位职责及操作
内容概要:本文介绍了稳定视频人脸修复(Stable Video Face Restoration, SVFR)方法,这是一个为解决一般化视频人脸修复(Generalized Video Face Restoration, GVFR)而设计的统一框架。作者针对现有技术未能有效处理时间一致性和运动伪影的问题提出SVFR方法,并通过实验展示其相较于单任务模型,在盲脸修复、颜色化和修补方面的性能显著提升。GVFR整合了三种子任务(BFR, colorization, 和 inpainting),并通过联合框架提高了训练效果,利用了预训练稳定视频扩散模型(SVD)。文中还引入了统一面部修复框架来确保跨子任务的一致特征表征,并提出了一种新的面部先验损失函数,通过面部特征点辅助修复过程并提高稳定性。 适用人群:适用于从事计算机视觉特别是视频识别和图像处理的研究者和技术工程师。 使用场景及目标:①用于高质量的人脸恢复,特别是在老旧电影或低质量监控录像等场合;②增强人脸识别准确性,确保长时间视频中身份一致性;③应用于需要真实感强的画面处理环境如在线会议直播。 其他说明:这项工作中提出的SVFR不仅解决了当前的技术难题,还为未来相关领域的研究和发展提供了重要参考价值,尤其体现在它所建立的新范例上,即如何通过多任务监督从有限数据集中获得更好的表现。
2025年义务教育新课程标准生物(2022年版)必考试题含答案.docx
直播带货销售业绩表
内容概要:本文详细记录了一个完整的基于Hadoop平台的WordCount任务实现过程,从环境准备到最终成果展示,涵盖了关键步骤的具体操作流程。首先介绍了创建所需文件夹结构并上传原始文本文件至HDFS;其次详述了构建Maven项目来组织相关源代码,以及定义Map(映射)、Combine(组合)、Reduce(归约)三个重要的处理环节所对应的程序逻辑;然后阐述了项目打包、分发过程及远程节点上部署运行该作业的整体思路;最后,通过访问Web界面确认最终生成的统计报告保存路径及其部分内容,验证任务成功完成。 适用人群:适用于初学者及有一定经验的数据工程师或研究人员,特别是那些希望快速掌握MapReduce模型实际应用技巧的人士。 使用场景及目标:此教程可以帮助用户深入了解Apache Hadoop生态系统内的MapReduce计算范式的运作机制。它演示了如何借助命令行工具高效管理和查询大规模非结构化或半结构化的数据集,从而支持后续更加复杂的分析任务的需求探索。此外,对于正在寻找入门级实战演练的学习者而言,这也是非常有价值的练习资料,既包括理论概念的学习也提供了充分的机会来进行动手实验。 其他说明:为了确保最佳实践效果,请注意跟随文中指引逐步尝试每一个新概念的应用,尤其是在编码部分,尽量不要跳过任何一步骤,并积极查阅官方文档或其他权威参考资料作为补充材料,遇到困难时也不必气馁,多做几次重复试验往往能带来意外收获。同时考虑到性能优化的可能性,可以在适当时候调整配置参数,比如增大堆栈容量或者更改块副本数目等。
2025义务教育历史新课程标准(2022版)必考题库含答案.docx
显示效果视频
员工晋升管理规定
2025年义务教育英语课程标准(2022版)必考题库及答案.docx
项目已获导师指导并通过的高分毕业设计项目,可作为课程设计和期末大作业,下载即用无需修改,项目完整确保可以运行。 包含:项目源码、数据库脚本、软件工具等,该项目可以作为毕设、课程设计使用,前后端代码都在里面。 该系统功能完善、界面美观、操作简单、功能齐全、管理便捷,具有很高的实际应用价值。 项目都经过严格调试,确保可以运行!可以放心下载 技术组成 语言:java 开发环境:idea 数据库:MySql8.0 部署环境:Tomcat(建议用 7.x 或者 8.x 版本),maven 数据库工具:navicat
风储联合系统:直驱风机与储能技术的完美融合,创新能源存储解决方案,风储联合系统 直驱风机加储能系统 ,核心关键词:风储联合系统; 直驱风机; 储能系统; 联合系统。,"风储联动:直驱风机与储能系统的绿色能源集成"
双馈风力发电机模型研究:Simulink中的完美波形效果展示,涵盖DFIG模型对风速变化与电流电压特性的精准模拟。,双馈风力发电机模型研究(DFIG),simulink模型。 给定风速变化,电流与电压等波形效果完美。 ,核心关键词:双馈风力发电机模型研究(DFIG); Simulink模型; 风速变化; 电流波形; 电压波形; 效果完美。,"双馈风力发电机Simulink模型研究:风速变化下的电流电压波形优化"