(原创)c++11改进我们的模式之改进表驱动模式
所谓表驱动法(Table-Driven Approach),简单讲是指用查表的方法获取值。表驱动是将一些通过较为复杂逻辑语句来得到数据信息的方式,通过查询表的方式来实现,将数据信息存放在表里。对于消除长的switch-case和if-else-if语句来说很有效,比如下面的代码:
string GetDayName(int day)
{
string dayName;
if(day==)
{
dayName="星期一";
}
else if(day==)
{
dayName="星期二";
}
else if(day==)
{
dayName="星期三";
}
else if(day==)
{
dayName="星期四";
}
else if(day==)
{
dayName="星期五";
}
else if(day==)
{
dayName="星期日";
}
}
这样的主要问题是:
- 代码太长,逻辑重复,圈复杂度高;
- 可维护性低,当新增一个流程分支时就要添加一个判断语句。
通过表驱动就一两条语句就可以代替上面长长的if-else语句。
string GetDayName(int day)
{
string dayNames[] = {"星期日","星期一","星期二","星期三","星期四","星期五","星期六"};
return dayNames[day]
}
这样不仅代码看起来简明,而且后面如果增加新的数值的话维护起来也更简单、方便。
表驱动法虽然对于消除长的if-else语句、提高代码质量很有用,但是一般的表驱动难以重用,即这个类的表驱动难以被其它类的表驱动所重用,因为其它类的表驱动逻辑和当前类的逻辑并不一定相同,具体来说是逻辑分支的函数不尽相同,这个问题导致表驱动模式的实现不够通用。
如果要做一个通用的表驱动模式就会面临两个问题,第一个问题是如何在表中注册各种类型的执行函数,这些执行函数形参不尽相同,用function是不行的,因为function的类型在一开始就定下来了;第二个问题是表找到了相应的执行函数之后调用该函数的问题,因为每个函数的形参不尽相同,如何以一种统一的方式去调用也是一个问题;问题二比较好解决,用c++11的可变模板参数即可;问题一需要通过类型擦除来解决,这要用到前面2.3.5节中实现的any做这个事情。c++11版本的表驱动模式,支持各种类型的key,执行函数支持普通函数、函数对象、lamda表达式和成员函数。
c++11之前写一个通用表驱动模式面临两个问题,第一个问题是如何在表中注册各种类型的执行函数,这些执行函数形参不尽相同,用function是不行的,因为function的类型在一开始就定下来了;第二个问题是表找到了相应的执行函数之后调用该函数的问题,因为每个函数的形参不尽相同,如何以一种统一的方式去调用也是一个问题;问题二比较好解决,用c++11的可变模板参数即可;问题一需要通过类型擦除来解决,这要用到boost::any做这个事情。c++11版本的表驱动模式,支持各种类型的key,执行函数支持普通函数、函数对象、lamda表达式和成员函数。
上代码:
#include <functional>
#include <type_traits>
template<typenameKey>
class TableDriver
{
public:
template<typename... Args, typenameFunc>
void Register(Key&&key, Func&&func)
{
//typedef decltype(std::declval<Func>()(std::declval<Args>()...)) rettype;
typedeftypename std::result_of<Func(Args...)>::typerettype;
auto f = std::function<rettype(Args&& ...)>([=](Args&& ... args){return func(std::forward<Args>(args)...); });
m_map[key] = f;
} template<typenameR = void, typename... Args>
R Execute(Key&&key, Args&&... args)
{
auto it = m_map.find(key);
if (it == m_map.end())
return (R)nullptr; auto f = it->second.AnyCast<std::function<R(Args&& ...)>>();
return f(std::forward<Args>(args)...);
} //不带参数的执行
template<typenameR>
R Execute(Key&&key)
{
auto it = m_map.find(key);
if (it == m_map.end())
return (R)nullptr; auto f = it->second.AnyCast<std::function<void()>>();
return f();
} private:
std::map<Key, Any> m_map;
};
测试代码:
struct MyStruct2
{
void g(int x)
{
cout<<x<<endl;
} void g1()
{
cout<<2<<endl;
}
}; struct MyStruct3
{
void operator()(int x) const
{
cout<< x +1<<endl;
}
}; void TestTbDriver()
{
TableDriver<int> dv;
auto f = [](int a, int b){cout<<a+b<<endl;};
auto fuc = [](int a){cout<<a<<endl;};
auto fuc1 = [](){cout<<1<<endl;}; dv.Register<decltype(f),int,int>(1, f); //带参数的lamda表达式
dv.Register<decltype(fuc), int>(2, fuc);
dv.SimpleRegister<decltype(fuc1)>(22, fuc1); //不带参数的lamda表达式 MyStruct2 st2;
dv.Register<int>(4, &st2, &MyStruct2::g); //带参数的成员函数
MyStruct3 st3;
dv.Register<decltype(st3), int>(11, st3); //函数对象
dv.SimpleRegister(44, &st2, &MyStruct2::g1); //不带参数的成员函数 int a=3,b=4;
dv.Execute(2, a);
dv.Execute(1, a, b);
dv.Execute(22);
dv.Execute(44);
dv.Execute(11, a);
}
structTdd
{
int Test(intx)
{
returnx + ;
}
}; void TestDriver()
{
TableDriver<string> dv;
dv.Register("aa", []{cout <<"aa test;"<< endl; });
dv.Execute("aa"); int y = ;
dv.Register<int, int>("aa", [](inta, intb){returna + b; });
auto t = dv.Execute<decltype(y)>("aa", , ); Tdd a;
dv.Register<int>("aa", [&a](intx){return a.Test(x); });
auto t1 = dv.Execute<int>("aa", ); dv.Register<string>("aa", [](stringx){returnx; });
auto t2 = dv.Execute<string>(std::move(string("aa")), std::move(string("test")));
}
测试结果:
/**********************做一个更新,对代码进行简化和优化,更少的代码,并且可以带返回值了 **********************/
代码中用到的Any就是博客:http://www.cnblogs.com/qicosmos/p/3420095.html 中的Any
#include "Any.hpp"
template<typename Key>
class TableDriver
{
public:
template<typename... Args, typename Func>
void Register(Key && key, Func&& func)
{
//typedef decltype(std::declval<Func>()(std::declval<Args>()...)) rettype;
typedef typename std::result_of<Func(Args...)>::type rettype;
auto f = std::function<rettype(Args && ...)>([=](Args && ... args){return func(std::forward<Args>(args)...); });
m_map[key] = f;
}
template<typename R = void, typename... Args>
R Execute(Key && key, Args&&... args)
{
auto it = m_map.find(key);
if (it == m_map.end())
return (R)nullptr;
auto f = it->second.AnyCast<std::function<R(Args && ...)>>();
return f(std::forward<Args>(args)...);
}
//不带参数的执行
template <typename R>
R Execute(Key && key)
{
auto it = m_map.find(key);
if (it == m_map.end())
return (R)nullptr;
auto f = it->second.AnyCast<std::function<void()>>();
return f();
}
private:
std::map<Key, Any> m_map;
};
测试代码:
struct Tdd
{
int Test(int x)
{
return x + 2;
}
};
void TestDriver()
{
TableDriver<string> dv;
dv.Register("aa", []{cout << "aa test;" << endl; });
dv.Execute("aa");
int y = 0;
dv.Register<int, int>("aa", [](int a, int b){return a + b; });
auto t = dv.Execute<decltype(y)>("aa", 3, 4);
Tdd a;
dv.Register<int>("aa", [&a](int x){return a.Test(x); });
auto t1 = dv.Execute<int>("aa", 3);
dv.Register<string>("aa", [](string x){return x; });
auto t2 = dv.Execute<string>(std::move(string("aa")), std::move(string("test")));
}
(原创)c++11改进我们的模式之改进表驱动模式的更多相关文章
- Tomcat内核之ASCII解码的表驱动模式
我们知道Tomcat通信是建立在Socket的基础上,而套接字在服务器端和客户端传递的报文都是未经过编码的字节流,每8位组成1个字节,计算机以二进制为基础,这是由于使用晶体管的开合状态表示1和0,这样 ...
- (原创)c++11改进我们的模式之改进访问者模式
本次讲c++11改进我们的模式之改进访问者模式 访问者模式是GOF23个设计模式中比较复杂的模式之一,但是它的功能也很强大,非常适合稳定的继承层次中对象的访问,可以在不修改被访问对象的情况下,动态添加 ...
- C++11在时空性能方面的改进
C++11在时空性能方面的改进 这篇我们聊聊C++11在时间和空间上的改进点: 主要包括以下方面: 新增的高效容器:array.forward_list以及unordered containers: ...
- [原创].NET 分布式架构开发实战五 Framework改进篇
原文:[原创].NET 分布式架构开发实战五 Framework改进篇 .NET 分布式架构开发实战五 Framework改进篇 前言:本来打算这篇文章来写DAL的重构的,现在计划有点改变.之前的文章 ...
- 【JavaScript】设计模式-module模式及其改进
写在前面 编写易于维护的代码,其中最重要的方面就是能够找到代码中重复出现的主题并优化他们,这也是设计模式最有价值的地方 说到这里...... <head first设计模式>里有一篇文章, ...
- 设计模式之module模式及其改进
写在前面 编写易于维护的代码,其中最重要的方面就是能够找到代码中重复出现的主题并优化他们,这也是设计模式最有价值的地方 <head first设计模式>里有一篇文章,是说使用模式的心智, ...
- 【原创实现】C 多线程入门Demo CAS Block 2种模式实现
分Cas和Block模式实现了demo, 供入门学习使用,代码全部是远程实现. 直接上代码: /* ================================================== ...
- Windows Azure Cloud Service (41) 修改云服务IIS托管管道模式为4.0经典模式
<Windows Azure Platform 系列文章目录> 这是笔者在之前的项目中遇到的问题,做一下总结,给网友做参考. 在一般情况下,Visual Studio开发的Cloud Se ...
- 微博feed系统的推(push)模式和拉(pull)模式和时间分区拉模式架构探讨
sns系统,微博系统都应用到了feed(每条微博或者sns里的新鲜事等我们称作feed)系统,不管是twitter.com或者国内的新浪微博,人人网等,在各种技术社区,技术大会上都在分享自己的feed ...
随机推荐
- UVa 10029 - Edit Step Ladders
題目:已知一些字典序排列的單詞,問能從中找到最大的一個有序單詞集合, 使得集合中的單詞每一個是有上一個單詞經過一次變換得來的(增.刪.改). 分析:dp,LIS.最大遞增子序列,不過數據較大须要優化. ...
- 【TP3.2.X】(同样适用于OT) 设置单入口index.php文件,区分PC/WAP/Wechat 三个终端
1.目的:本教程适用于 OneThink1.0.或者TP3.2.X 系列,达到单入口index.php文件,区分PC/WAP/Wechat 三个终端 2.启发至 : http://www.thinkp ...
- 【laravel5.*】添加ide_helper.php 助手
1.参照文档:https://github.com/barryvdh/laravel-ide-helper#automatic-phpdoc-generation-for-laravel-facade ...
- Android 圆角输入框
draweable文件下建立一个名字为shape的XML文件: <shape xmlns:android="http://schemas.android.com/apk/r ...
- Using PHP as a Spring MVC View via Quercus(转)
原贴: http://blog.caucho.com/2009/04/14/using-php-as-a-spring-mvc-view-via-quercus/ This week, I’ve be ...
- java struts2入门学习--防止表单重复提交.OGNL语言学习
一.知识点回顾 防止表单重复提交核心思想: 客户端和服务器端和写一个token,比较两个token的值相同,则非重复提交;不同,则是重复提交. 1.getSession三种方式比较: request. ...
- JavaScript 表单验证入门
JavaScript 表单验证 JavaScript 可用来在数据被送往服务器前对 HTML 表单中的这些输入数据进行验证. 被 JavaScript 验证的这些典型的表单数据有: 用户是否已填写表单 ...
- SIGTERM等信号含义【转】
主要是做sigterm 和sigkill区别的比较,之前的好多操作,在结束正在运行的一个程序的时候,常用kill,这个以后的注意,在sigterm不起作用的时候,再使用kill; 原文地址:http: ...
- nginx端合并JS
转载于:http://www.tuicool.com/articles/IjMNnq 配置文件: location /static/css/ { concat on; concat_max_files ...
- IP概念盛行的背后:资本在狂欢,电影想哭泣 IP,英文“Intellectual Property”的缩写,直译为“知识产权”。它的存在方式很多元,可以是一个故事,也可以是某一个形象,运营成功的IP可以在漫画、小说、电影、玩具、手游等不同的媒介形式中转换。
IP概念盛行的背后:资本在狂欢,电影想哭泣 IP容易拉投资.谈合作,甚至还能简化宣发途径,越来越多的人涌入了电影这个产业,争抢IP成为他们进入行业的最快捷的方法.IP盛行暴露出的另一个问题是国产电影原 ...