【学习笔记】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天更新一章,主要是给大家分享一下,我所学到的知识,如果有什么错误请在评论中指点出来,我一定虚心接受,那么废话不多说开始我们今天的学习分享吧! 前几 ...
随机推荐
- 【笔记】CF1607F Robot on the Board 2 及相关
题目传送门 记忆化搜索 首先,这题 \(10000\) 组 \(2000\times 2000\) 的数据直接爆搜肯定会超时.想到,如果一个点的答案已经被更新过,之后走到这个点能再多走的点也就确定了, ...
- RabbitMq发布确认
RabbitMq发布确认 发布确认原理 生产者将信道设置成 confirm 模式,一旦信道进入 confirm 模式,所有在该信道上面发布的消息都将会被指派一个唯一的 ID(从 1 开始),一旦消息被 ...
- JAVA 用分苹果来理解本题
思路 其实这是一道非常经典的分苹果问题:有m个一样的苹果和n个一样的盘子,把苹果放盘子里,每个盘子允许0-m个苹果,求问有多少种分法? 与本题的共通之点在于,输入的正整数可以看成m个苹果,拆分出的加数 ...
- ftp多用户多目录配置
测试环境:centos7 1. 装包与卸载 yum -y install vsftpd yum -y autoremove vsftpd&&rm -rf /etc/vsftpd /et ...
- 1759D(数位变0)
题目链接 题目大意: 给你两个整数n, m.你需要求一个数,它满足如下条件: 是n的整数倍,且倍数小于m. 你应该使其末尾的0尽可能的多(如100后面有2个零,1020后面有一个零,我们应该输出100 ...
- python(牛客)试题解析3 - 困难
导航 一.找到已经最大承重的背包内如何放入最大价值的物品的最优解 二.查找一个字符串中包含另外一个字符串(可打乱顺序)的次数三.计算正整数数组从头走到最后一个成员所需的最小步骤四.计算字符串非严格递增 ...
- 痞子衡嵌入式:MCUXpresso IDE下高度灵活的FreeMarker链接文件模板机制
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是MCUXpresso IDE下高度灵活的FreeMarker链接文件模板机制. 痞子衡之前写过一篇文章 <MCUXpresso I ...
- jmeter 从多个数中随机取一个值的方法
问题描述:使用jmeter进行接口测试时,遇到枚举值(如:10代表闲置.15代表使用中.20代表维修等)我们需要随机取一个类型传到接口中. 解决思路:通过函数助手查找随机函数,找到__chooseRa ...
- 打印九九乘法表,打印金字塔-java
/** * *** ***** 打印如图金字塔 *=i*2-1 (竖)空格数=列数-1 */ public class Circulate{ public static void main(Strin ...
- <一>智能指针基础
代码1 int main(){ //裸指针,手动开辟,需要自己释放,如果忘记了或者因为 //程序逻辑导致p没有释放,那么就会导致内存泄漏 int *p=new int(10); if(***){ re ...