使用std::function改善模板的低效性
泛型编程中,模板会根据传入类型的不同,生成多种实例,相对低效。
模板编程:
#include <iostream> using namespace std; //未使用函数包装器
template <typename T,typename F>
T use_f(T v, F f)
{
static int count = ;
count++;
cout<<"use_f count = "<<count<<": &count"<<&count<<endl;
return f(v);
} class Fp
{
private:
double z_;
public:
Fp(double z = 1.0):z_(z){}
double operator ()(double p) {return z_*p;}
}; class Fq
{
private:
double z_;
public:
Fq(double z = 1.0):z_(z){}
double operator ()(double p) {return z_+p;}
}; double dub(double x){return 2.0*x;}
double square(double x){return x*x;} int main(int argc, char *argv[])
{
double y = 1.2;
cout<<"Function pointer dub:"<<endl;
cout<<" --"<<use_f(y,dub);
cout<<"Function pointer square:"<<endl;
cout<<" --"<<use_f(y,square);
cout<<"Function pointer Fp:"<<endl;
cout<<" --"<<use_f(y,Fp());
cout<<"Function pointer Fq:"<<endl;
cout<<" --"<<use_f(y,Fq());
cout<<"Function pointer lambada1:"<<endl;
cout<<" --"<<use_f(y,[](double u){return u*u;});
cout<<"Function pointer lambada2:"<<endl;
cout<<" --"<<use_f(y,[](double u){return u+2.5;});
return ;
}

模板编程的优势是高度抽象,将算法统一封装,但是其根据类型实例化的特性也造成了其低效性。
以上执行结果,通过查看静态模板函数中,静态数据的地址,其产生了5个函数实例。代码量受传入类型的影响产生了巨大的增量。
通过std::function来减少模板的实例化:
#include <iostream>
#include <functional>
using namespace std; //使用函数包装器
template <typename T,typename F>
T use_f(T v, F f)
{
static int count = ;
count++;
cout<<"use_f count = "<<count<<": &count"<<&count<<endl;
return f(v);
} class Fp
{
private:
double z_;
public:
Fp(double z = 1.0):z_(z){}
double operator ()(double p) {return z_*p;}
}; class Fq
{
private:
double z_;
public:
Fq(double z = 1.0):z_(z){}
double operator ()(double p) {return z_+p;}
}; double dub(double x){return 2.0*x;}
double square(double x){return x*x;} int main(int argc, char *argv[])
{
double y = 1.2;
// cout<<"Function pointer dub:"<<endl;
// cout<<" --"<<use_f(y,dub);
// cout<<"Function pointer square:"<<endl;
// cout<<" --"<<use_f(y,square);
// cout<<"Function pointer Fp:"<<endl;
// cout<<" --"<<use_f(y,Fp());
// cout<<"Function pointer Fq:"<<endl;
// cout<<" --"<<use_f(y,Fq());
// cout<<"Function pointer lambada1:"<<endl;
// cout<<" --"<<use_f(y,[](double u){return u*u;});
// cout<<"Function pointer lambada2:"<<endl;
// cout<<" --"<<use_f(y,[](double u){return u+2.5;}); std::function<double(double)> fdub = dub;
std::function<double(double)> fsquare = square;
std::function<double(double)> fFp = Fp();
std::function<double(double)> fFq = Fq();
std::function<double(double)> lambada1 = [](double u){return u*u;};
std::function<double(double)> lambada2 = [](double u){return u+2.5;}; cout<<"Function pointer dub:"<<endl;
cout<<" --"<<use_f(y,fdub);
cout<<"Function pointer square:"<<endl;
cout<<" --"<<use_f(y,fsquare);
cout<<"Function pointer Fp:"<<endl;
cout<<" --"<<use_f(y,fFp);
cout<<"Function pointer Fq:"<<endl;
cout<<" --"<<use_f(y,fFq);
cout<<"Function pointer lambada1:"<<endl;
cout<<" --"<<use_f(y,lambada1);
cout<<"Function pointer lambada2:"<<endl;
cout<<" --"<<use_f(y,lambada2); return ;
}

所有的静态变量只有一个地址,也就是说模板 函数只产生了一个实例,其类型只需要匹配是一个std::function<double(double)>传入的对象即可,模板的效率大为提高。
但是看来好像代码量并未减少甚至还有些许增多,下面我们通过优化来解决这个问题。
熊掌与鱼:
#include <iostream>
#include <functional>
using namespace std; //模板中使用函数包装器
template <typename T>
T use_f(T v, std::function<T(T)> f)
{
static int count = ;
count++;
cout<<"use_f count = "<<count<<": &count"<<&count<<endl;
return f(v);
} class Fp
{
private:
double z_;
public:
Fp(double z = 1.0):z_(z){}
double operator ()(double p) {return z_*p;}
}; class Fq
{
private:
double z_;
public:
Fq(double z = 1.0):z_(z){}
double operator ()(double p) {return z_+p;}
}; double dub(double x){return 2.0*x;}
double square(double x){return x*x;} int main(int argc, char *argv[])
{
double y = 1.2;
//这里需要<double>使得std::function<T(T)>实例化为具体的对象
cout<<"Function pointer dub:"<<endl;
cout<<" --"<<use_f<double>(y,&dub);
cout<<"Function pointer square:"<<endl;
cout<<" --"<<use_f<double>(y,square);
cout<<"Function pointer Fp:"<<endl;
cout<<" --"<<use_f<double>(y,Fp());
cout<<"Function pointer Fq:"<<endl;
cout<<" --"<<use_f<double>(y,Fq());
cout<<"Function pointer lambada1:"<<endl;
cout<<" --"<<use_f<double>(y,[](double u){return u*u;});
cout<<"Function pointer lambada2:"<<endl;
cout<<" --"<<use_f<double>(y,[](double u){return u+2.5;}); return ;
}

实例化与代码量都减少了,鱼和熊掌兼得。
使用std::function改善模板的低效性的更多相关文章
- 用lambda表达式和std::function类模板改进泛型抽象工厂设计
- 剖析std::function接口与实现
目录 前言 一.std::function的原理与接口 1.1 std::function是函数包装器 1.2 C++注重运行时效率 1.3 用函数指针实现多态 1.4 std::function的接 ...
- C++11中的std::function
看看这段代码 先来看看下面这两行代码: std::function<void(EventKeyboard::KeyCode, Event*)> onKeyPressed; std::fun ...
- 【转】C++11中的std::function
原文地址:http://www.jellythink.com/archives/771 看看这段代码 先来看看下面这两行代码: std::function<void(EventKeyboard: ...
- 第12课 std::bind和std::function(3)_std::function可调用对象包装器
1. std::function (1)首先是一个类模板,用于包装可调用对象.可以容纳除了类成员(函数)指针之外的所有可调用对象. (2)可以将普通函数,lambda表达式和函数对象类统一起来.尽管它 ...
- 第10课 std::bind和std::function(1)_可调用对象
1. 几种可调用对象(Callable Objects) (1)普通函数指针或类成员的函数指针 (2)具有operator()成员函数的类对象(仿函数).如c++11中的std::function类模 ...
- std::function介绍 -转载
类模版std::function是一种通用.多态的函数封装.std::function的实例可以对任何可以调用的目标实体进行存储.复制.和调用操作,这些目标实体包括普通函数.Lambda表达式.函数指 ...
- callable object与新增的function相关 C++11中万能的可调用类型声明std::function<...>
在c++11中,一个callable object(可调用对象)可以是函数指针.lambda表达式.重载()的某类对象.bind包裹的某对象等等,有时需要统一管理一些这几类对象,新增的function ...
- C++11 std::function用法
转自 http://www.hankcs.com/program/cpp/c11-std-function-usage.html function可以将普通函数,lambda表达式和函数对象类统一起来 ...
随机推荐
- 编写并提取通用 ShellCode
简易 ShellCode 虽然可以正常被执行,但是还存在很多的问题,因为上次所编写的 ShellCode 采用了硬编址的方式来调用相应API函数的,那么就会存在一个很大的缺陷,如果操作系统的版本不统一 ...
- Qt里的原子操作QAtomicInteger,有挑战性,使用Q_ATOMIC_INT{nn}_IS_SUPPORTED测试系统是否支持
所谓原子操作,即一系列复杂的操作能一气呵成,中间不被其他的操作打断.这在多线程程序中尤其常见,但要实现这种功能,既要考虑程序的良好设计,又要关心特定平台的体系结构和相关编译器对原子特性的支持程度.所以 ...
- dockerfile相关命令
官方dockerfile:https://github.com/play-with-docker/play-with-docker 可以根据一直的镜像,学习dockerfile的编写 dockerfi ...
- (一)Maven基础及第一个Maven工程
一.Maven介绍 ANT/Maven/gradle是一个项目管理工具,它包含了一项目对象模型(Project Object Model),一组标准集合,一个项目生命周期(Project Lifecy ...
- springboot中使用验证码kaptcha
1.pom.xml引入kaptcha所需要的jar包 <!-- 验证码 --> <dependency> <groupId>com.github.penggle&l ...
- Effective Java 读书笔记(二):对象通用方法
1 重写equals方法时请遵守通用约定 (1)无需覆盖equals方法的情况 要求独一无二 不要求逻辑相等 超类已经覆盖equals方法,对其子类也适用 一个类是私有的或者是包私有(可以重写后抛出异 ...
- 把一个质量为M0 的物体加速到 时间慢100倍 需要多大能量
洛伦兹系数 γ = 100: 运动时候的质量 M = γM0 速度 v = c*sqrt(1-1/γ2) E = 1/2 * m0c2 *γ* (1-1/γ2) 如果一个60kg 的人想要加 ...
- python常见函数运用【一】
1.Python hasattr() 函数 描述hasattr() 函数用于判断对象是否包含对应的属性. 语法 hasattr 语法: hasattr(object, name)参数object -- ...
- Element-ui-Basic
一.Layout 布局 1.基础布局 <el-row> <el-col :span="24"><div class="grid-conten ...
- vue去哪儿网项目环境配置
一.首先安装node.js 根据自己的(windows或mac)系统进行安装node,在开发环境中一般安装LTS版本.安装成功后,在终端输入"node -v"和"npm ...