介绍说明

模板设计模式是一种非常简单的设计模式,其主要是利用了虚函数的特性实现。非常适合应用在一些算法、流程、业务逻辑是固定的形式,其中某些步骤的实现方式又无法确定下来的场景。

举例说明

以下为模拟某芯片基于串口通信的固件升级代码,可以提供一个 FirmwareUpgrade 的类用于使用者集成。考虑到跨平台,该类的串口操作接口声明为纯虚函数,由使用者自己根据所用平台实现,而不需要关注具体的升级细节。

class FirmwareUpgrade
{
protected: // 具体的串口操作实现延迟到子类实现
// 在不同的平台下, 提供对应平台的串口操作实现即可
virtual bool openSerialPort(int nBaudRate, int nDataBit, int nStopBit, int nParityBit) = 0;
virtual bool wrtieSerialPort(void *pOutBuffer, size_t nLength) = 0;
virtual size_t readSerialPort(void *pInBuffer, size_t size) = 0;
virtual bool closeSerialPort() = 0; public: FirmwareUpgrade() {};
virtual ~FirmwareUpgrade() {}; // 开始升级
bool upgrade(const char *pFirmwareFile) { init();
...
enterUpgradeMode();
...
erasePartition();
...
updatePartition(pFirmwareFile);
...
verifyPartition();
...
resetDevice();
...
release(); return true;
} private: bool sendCommand(int cmd, void *pdata, size_t size) { if (!wrtieSerialPort(package, sizeof(package))) {
return false;
} if(readSerialPort(package, sizeof(package))) {
return false;
} return package->done;
} bool init(){
...
if (!openSerialPort(115200, 8, 1, 0)) {
return false;
}
...
return true;
} bool enterUpgradeMode(){
...
} bool erasePartition(){
...
} bool updatePartition(const char *pFirmwareFile){
FILE fp = fopen(pFirmwareFilel, "rb");
...
while (fread(buffer, sizeof(buffer), 1, fp) == 1)
{
if (!sendCommand(update_COMMAND, NULL, 0)) {
fclose(fp);
return false;
}
}
...
fclose(fp);
return true;
} bool verifyPartition() {
...
} bool resetDevice(){
...
} bool release() {
...
return closeSerialPort();
}
};

如 Windows 平台开发者,只需要利用  Windows 平台下接口实现串口的打开、写入、读取、关闭四个接口即可:


class FirmwareUpgradeWin : public FirmwareUpgrade
{
private: virtual bool openSerialPort(int nBaudRate, int nDataBit, int nStopBit, int nParityBit)
{ com_handle = CreateFile("//.//COM9", GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
if (com_handle != INVALID_HANDLE_VALUE){
...
}
...
} virtual bool wrtieSerialPort(void *pOutBuffer, size_t nLength)
{
if (com_handle != INVALID_HANDLE_VALUE)
{
DWORD NumberOfBytesWrite;
if (!WriteFile(com_handle, buffer, length, &NumberOfBytesWrite, NULL))
{
DWORD error = GetLastError();
printf("error:%d\n", error);
}
return NumberOfBytesWrite;
}
} virtual size_t readSerialPort(void *pInBuffer, size_t size)
{
if (com_handle != INVALID_HANDLE_VALUE)
{
DWORD NumberOfBytesRead;
ReadFile(com_handle, buffer, length, &NumberOfBytesRead, NULL);
return NumberOfBytesRead;
}
return 0;
} virtual bool closeSerialPort()
{
// 关闭硬件接口
if (com_handle != INVALID_HANDLE_VALUE)
{
CloseHandle(com_handle);
}
} public bool upgrade(const char *pFirmwareFile) {
return FirmwareUpgrade::upgrade(pFirmwareFile);
}
}; int main()
{
FirmwareUpgradeWin mFirmwareUpgrade; mFirmwareUpgrade.upgrade("d:\\xxxx.bin"); return 0;
}

如 Linux 平台下的实现:

class FirmwareUpgradeLinux : public FirmwareUpgrade
{
private: virtual bool openSerialPort(int nBaudRate, int nDataBit, int nStopBit, int nParityBit)
{ m_fd = ::open(device_node, O_RDWR | O_NOCTTY);
....
tcgetattr(m_fd, &m_options);
....
return true;
} virtual bool wrtieSerialPort(void *pOutBuffer, size_t nLength)
{
...
byte = write(m_fd, pOutBuffer, nLength));
...
return true;
} virtual size_t readSerialPort(void *pInBuffer, size_t size)
{
...
select(m_fd + 1, &fs_read, NULL, NULL, &time);
...
byte = read(m_fd, data, size);
...
return byte;
} virtual bool closeSerialPort()
{
...
close(m_fd);
...
return true
} public: bool upgrade(const char *pFirmwareFile) {
return FirmwareUpgrade::upgrade(pFirmwareFile);
}
}; int main()
{
FirmwareUpgradeLinux mFirmwareUpgrade; mFirmwareUpgrade.upgrade("d:\\xxxx.bin"); return 0;
}

总结说明

主要就是将差异化的抽离出来,延迟到子类实现,而固定的逻辑处理则由父类封装并提供接口。相当于定义一个模板,子类只要按照这个模板实现相应的接口,由父类反向调用子类所实现的接口,来完成具体的逻辑功能。这样使用者不需要关心具体的升级逻辑实现,只需要按照要求完成相应的接口,即可使用升级功能,从而降低复杂性,又提升了灵活性。

【学习笔记】C/C++ 设计模式 - 模板模式的更多相关文章

  1. 【设计模式】Java设计模式 - 模板模式

    Java设计模式 - 模板模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 原创作品,更多关注我CSDN: 一个有梦有戏的人 准备将博客园.CSDN一起记录分享自己 ...

  2. 12. 星际争霸之php设计模式--模板模式

    题记==============================================================================本php设计模式专辑来源于博客(jymo ...

  3. Django 学习笔记(五)模板标签

    关于Django模板标签官方网址https://docs.djangoproject.com/en/1.11/ref/templates/builtins/ 1.IF标签 Hello World/vi ...

  4. Django 学习笔记(四)模板变量

    关于Django模板变量官方网址:https://docs.djangoproject.com/en/1.11/ref/templates/builtins/ 1.传入普通变量 在hello/Hell ...

  5. Django 学习笔记(三)模板导入

    本章内容是将一个html网页放进模板中,并运行服务器将其展现出来. 平台:windows平台下Liunx子系统 目前的目录: hello ├── manage.py ├── hello │ ├── _ ...

  6. 并发编程学习笔记(9)----AQS的共享模式源码分析及CountDownLatch使用及原理

    1. AQS共享模式 前面已经说过了AQS的原理及独享模式的源码分析,今天就来学习共享模式下的AQS的几个接口的源码. 首先还是从顶级接口acquireShared()方法入手: public fin ...

  7. Redis学习笔记八:集群模式

    作者:Grey 原文地址:Redis学习笔记八:集群模式 前面提到的Redis学习笔记七:主从复制和哨兵只能解决Redis的单点压力大和单点故障问题,接下来要讲的Redis Cluster模式,主要是 ...

  8. 再起航,我的学习笔记之JavaScript设计模式30(简单模板模式)

    简单模板模式 概念介绍 简单模板模式(Simple template): 通过格式化字符串拼凑出视图避免创建视图时大量节点操作,优化内存开销. 创建模板 在实际的业务中如果我们需要进行前后台交互,或多 ...

  9. 再起航,我的学习笔记之JavaScript设计模式09(原型模式)

    我的学习笔记是根据我的学习情况来定期更新的,预计2-3天更新一章,主要是给大家分享一下,我所学到的知识,如果有什么错误请在评论中指点出来,我一定虚心接受,那么废话不多说开始我们今天的学习分享吧! 我们 ...

  10. 再起航,我的学习笔记之JavaScript设计模式05(简单工程模式)

    我的学习笔记是根据我的学习情况来定期更新的,预计2-3天更新一章,主要是给大家分享一下,我所学到的知识,如果有什么错误请在评论中指点出来,我一定虚心接受,那么废话不多说开始我们今天的学习分享吧! 前几 ...

随机推荐

  1. h5 websocket 断开重新连接

    最近的项目中使用ws 长连接来接收和发送消息, 直接上代码 import * as SockJS from "sockjs-client"; import Stomp from & ...

  2. 如何理解Java中眼花缭乱的各种并发锁?

    在互联网公司面试中,很多小伙伴都被问到过关于锁的问题. 今天,我给大家一次性把Java并发锁的全家桶彻底讲明白.包括互斥锁.读写锁.重入锁.公平锁.悲观锁.自旋锁.偏向锁等等等等.视频有点长,大家一定 ...

  3. EventBridge助力阿里云视觉智能开放平台AI智能存储实践

    本文作者:李建,阿里巴巴达摩院技术专家. 01 视觉智能开放平台(VIAPI)业务场景介绍 阿里云视觉智能开放平台(简称 VIAPI),是基于之前很多技术实践经验积累的 AI 能力的沉淀平台.目前整个 ...

  4. 渗透技巧基于Swagger-UI的XSS

    目录 免责声明: 漏洞简述: 漏洞实现 POC 漏洞利用 如何大规模找到 Swagger UI Google FOFA XRAY 修复 免责声明:   本文章仅供学习和研究使用,严禁使用该文章内容对互 ...

  5. 让 Serverless 更普惠,阿里云函数计算 FC 宣布全面降价,最大幅度达 37.5%

    11月5日,2022 杭州 · 云栖大会上,阿里云宣布函数计算 FC 开启全面降价,vCPU 单价降幅** 11%,其他的各个独立计费项最高降幅达 37.5%**. 本次云栖大会上,阿里云智能总裁张建 ...

  6. 带你从入门到精通学习WireShark

    个人名片: 因为云计算成为了监控工程师‍ 个人博客:念舒_C.ying CSDN主页️:念舒_C.ying 带你从入门到精通学习WireShark 一.什么是WireShark? 二.WireShar ...

  7. go slice不同初始化方式性能&数组比较

    go语言开发中,slice是我们常用的数据类型之一,也是因为它的灵活性,自己也很少使用数组,当然我也知道它的一些特性,不过没有真实的去验证它,因为大多数使用场景没必要对code太过苛刻,但是如果封装作 ...

  8. orcl rollup 分组小计、合计

    表数据: select * from group_test; 分组小计.合计: select group_id, decode(concat(job, group_id), null, '合计', g ...

  9. 【企业流行新数仓】Day03:SuperSet图表,Ranger权限、脱敏、行级别过滤,Atlas元数据、查询和查看全表/字段血缘依赖,Zabbix告警

    一.SuperSet-图表展示 1.概念 (1)概念 通过dashboard(仪表盘)对图表中的数据进行展示 BI工具:根据配置的要求,进行数据源的配置即可 是准商业级别的BI web应用 (2)原理 ...

  10. JS传值与应用

    问题提出:在进行页面书写的时候,有时候需要进行动态页面拼接,在动态拼接的时候,涉及到函数的调用,函数的传值可能是HTML标签,或者含有json的标签,这样在传值时就有可能出现问题,由于"&q ...