【学习笔记】C/C++ 设计模式 - 模板模式
介绍说明
模板设计模式是一种非常简单的设计模式,其主要是利用了虚函数的特性实现。非常适合应用在一些算法、流程、业务逻辑是固定的形式,其中某些步骤的实现方式又无法确定下来的场景。
举例说明
以下为模拟某芯片基于串口通信的固件升级代码,可以提供一个 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++ 设计模式 - 模板模式的更多相关文章
- 【设计模式】Java设计模式 - 模板模式
Java设计模式 - 模板模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 原创作品,更多关注我CSDN: 一个有梦有戏的人 准备将博客园.CSDN一起记录分享自己 ...
- 12. 星际争霸之php设计模式--模板模式
题记==============================================================================本php设计模式专辑来源于博客(jymo ...
- Django 学习笔记(五)模板标签
关于Django模板标签官方网址https://docs.djangoproject.com/en/1.11/ref/templates/builtins/ 1.IF标签 Hello World/vi ...
- Django 学习笔记(四)模板变量
关于Django模板变量官方网址:https://docs.djangoproject.com/en/1.11/ref/templates/builtins/ 1.传入普通变量 在hello/Hell ...
- Django 学习笔记(三)模板导入
本章内容是将一个html网页放进模板中,并运行服务器将其展现出来. 平台:windows平台下Liunx子系统 目前的目录: hello ├── manage.py ├── hello │ ├── _ ...
- 并发编程学习笔记(9)----AQS的共享模式源码分析及CountDownLatch使用及原理
1. AQS共享模式 前面已经说过了AQS的原理及独享模式的源码分析,今天就来学习共享模式下的AQS的几个接口的源码. 首先还是从顶级接口acquireShared()方法入手: public fin ...
- Redis学习笔记八:集群模式
作者:Grey 原文地址:Redis学习笔记八:集群模式 前面提到的Redis学习笔记七:主从复制和哨兵只能解决Redis的单点压力大和单点故障问题,接下来要讲的Redis Cluster模式,主要是 ...
- 再起航,我的学习笔记之JavaScript设计模式30(简单模板模式)
简单模板模式 概念介绍 简单模板模式(Simple template): 通过格式化字符串拼凑出视图避免创建视图时大量节点操作,优化内存开销. 创建模板 在实际的业务中如果我们需要进行前后台交互,或多 ...
- 再起航,我的学习笔记之JavaScript设计模式09(原型模式)
我的学习笔记是根据我的学习情况来定期更新的,预计2-3天更新一章,主要是给大家分享一下,我所学到的知识,如果有什么错误请在评论中指点出来,我一定虚心接受,那么废话不多说开始我们今天的学习分享吧! 我们 ...
- 再起航,我的学习笔记之JavaScript设计模式05(简单工程模式)
我的学习笔记是根据我的学习情况来定期更新的,预计2-3天更新一章,主要是给大家分享一下,我所学到的知识,如果有什么错误请在评论中指点出来,我一定虚心接受,那么废话不多说开始我们今天的学习分享吧! 前几 ...
随机推荐
- Python基础之模块:5、 第三方模块 requests模块 openpyxl模块
目录 一.第三方模块的下载与使用 1.什么是第三方模块 2.如何安装第三方模块 方式一:pip工具 方式二:pycharm中下载 3.注意事项 1.报错并有警告信息 2.报错,提示关键字 3.报错,无 ...
- Python基础之函数:2、globlal与nonlocal和闭包函数、装饰器、语法糖
目录 一.global与nonlocal 1.global 2.nonlocal 二.函数名的多种用法 三.闭包函数 1.什么是闭包函数 2.闭包函数需满足的条件 3.闭包函数的作用 4.闭包函数的实 ...
- Qt--无边框窗口完美(FrameLess)实现,包含缩放和移动功能重写。
前言 Qt原本的窗口虽然可以通过QSS样式进行美化,但是只是对客户区有用,对于客户区是无效的.所以想做出一个比较好看的程序,还得自己重写实现无边框窗口. Qt实现无边框其实一句代码就可以,但是窗口自带 ...
- 【深入浅出 Yarn 架构与实现】1-2 搭建 Hadoop 源码阅读环境
本文将介绍如何使用 idea 搭建 Hadoop 源码阅读环境.(默认已安装好 Java.Maven 环境) 一.搭建源码阅读环境 一)idea 导入 hadoop 工程 从 github 上拉取代码 ...
- java安全之CC1浅学(1)
前言 由于CC链还是比较复杂的,我们可以先看命令执行的部分payload之后再加上反序列化部分组成一个完整的payload 调试一 项目导入依赖,这里使用3.1版本 <!-- https://m ...
- .NET 7 AOT 的使用以及 .NET 与 Go 互相调用
目录 背景 C# 部分 环境要求 创建一个控制台项目 体验 AOT 编译 C# 调用库函数 减少体积 C# 导出函数 C# 调用 C# 生成的 AOT Golang 部分 安装 GCC Golang ...
- MessagePack 和System.Text.Json 序列号 反序列化对比
本博客将测试MessagePack 和System.Text.Json 序列号 反序列化性能 项目文件: Program.cs代码: using BenchmarkDotNet.Running; us ...
- 聊聊Go里面的闭包
以前写 Java 的时候,听到前端同学谈论闭包,觉得甚是新奇,后面自己写了一小段时间 JS,虽只学到皮毛,也大概了解到闭包的概念,现在工作常用语言是 Go,很多优雅的代码中总是有闭包的身影,看来不了解 ...
- C++一个吃豆人小游戏
C++一个吃豆人小游戏 代码如下 #include <cstdio>#include <iostream>#include <ctime>#include < ...
- 我开发的开源项目,让.NET7中的EFCore更轻松地使用强类型Id
在领域驱动设计(DDD)中,有一个非常重要的概念:"强类型Id".使用强类型Id来做标识属性的类型会比用int.Guid等通用类型能带来更多的好处.比如有一个根据根据Id删除用户的 ...