1. optional类的实现

(1)optional的功能

  ①optional<T>的内部存储空间可能存储了T类型的值,也可能没有。只有当optional被T初始化之后,这个optional才是有效的。否则是无效的。它实现了未初始化的概念

  ②optional可以用于解决函数返回无效值的问题。当函数返回一个未初始化的Optional对象时,表明函数正确执行了,只是结果不是有用的值。

  ③举例:optional<int> op; //未被初始化。 optional<int> op = 1; //初始化。

(2)实现细节

  ①由于optional<T>需要容纳T的值,所以需要一个缓冲区来保存它,但考虑到内存对齐,需要将T指定在对齐的位置上。可以通过std::alignment_of <T>::value来获取T的内存对齐大小。并通过std::aligned_storage<sizeof(T), aligned(T)>来定义T的内存对齐类型(该模板的本质就重新定义T经对齐后的一种新类型)。

template<unsigned size, unsigned alignment>
struct aligned_storage
{
using type = struct { alignas(alignment) unsigned char data[size]; };
};

  ②std::aligned_storage一般和placement_new结合使用(见Optional类的create函数),用于初始化由std::aligned_storage定义的一片内存空间。

  ③增加一个m_hasInit标记来记录T空间是否己经初始化。

【编程实验】Optional类的实现

//Optional.hpp

#ifndef _OPTIONAL_H_
#define _OPTIONAL_H_ #include <type_traits>
#include <utility> //for std::forward
#include <stdexcept> template <typename T>
class Optional
{
//std::alignment_of<T>::value获取T的内存对齐大小,std::aligned_storage将T重定义为对齐后的类型
using data_t = typename std::aligned_storage<sizeof(T), std::alignment_of<T>::value>::type;
private:
data_t m_data; //内存对齐缓冲区
bool m_hasInit; //是否己经初始化
private:
//调用placement_new来创建T对象
template<class... Args>
void create(Args... args) //可以接收左右值
{
new (&m_data) T(std::forward<Args>(args)...); //调用T的构造函数来初始化m_data空间
m_hasInit = true;
} //销毁缓冲区的对象
void destroy()
{
if(m_hasInit){
m_hasInit = false;
((T*)(&m_data))->~T(); //调用T的析构函数
}
} //缓冲区的拷贝
void copy(const data_t& val)
{
destroy();
new (&m_data) T(*((T*)(&val)));
} //缓冲区的移动
void move(data_t&& val)
{
destroy(); //调用T的移动构造函数进行移动
new (&m_data) T(std::move(*((T*)(&val))));
} //Optional赋值操作(左值版本)
void assign(const Optional& other)
{
if(other.isInit()){
copy(other.m_data);
m_hasInit = true;
}else{
destroy();
}
} //Optional赋值操作(右值版本)
void assign(Optional&& other)
{
if(other.isInit()){
move(std::move(other.m_data));
m_hasInit = true; other.destroy(); //other失去资源控制权
}else{
destroy();
}
} public:
Optional():m_hasInit(false){};
Optional(const T& v)
{
create(v);
} Optional(T&& v) : m_hasInit(false)
{
create(std::move(v));
} Optional(const Optional& other) : m_hasInit(false)
{
if(other.isInit()){
assign(other);
}
} Optional(Optional&& other) : m_hasInit(false)
{
if(other.isInit()){
assign(std::move(other));
other.destroy();
}
} //根据参数创建对象,如emplace(1,2);
template<class ...Args>
void emplace(Args&& ...args)
{
destroy();
create(std::forward<Args>(args)...);
} Optional& operator=(const Optional& other)
{
assign(other);
return *this;
} Optional& operator=(Optional&& other)
{
assign(std::move(other));
return *this;
} explicit operator bool() const //类型转换函数,如if(op)
{
return isInit();
} T& operator*()
{
if(isInit()){
return *((T*)(&m_data));
} throw std::logic_error{"try to get data in a Optional which is not initialized"};
} const T& operator*() const
{
if(isInit()){
return *((T*)(&m_data));
} throw std::logic_error{"try to get data in a Optional which is not initialized"};
} T* operator->()
{
return &operator*();
} const T* operator->() const
{
return &operator*();
} bool operator==(const Optional<T>& rhs) const
{
bool bInit = bool(*this);
return ( !bInit != (!rhs) ? //*this和rhs中一个初始化,一个未初始化
false :
!bInit ? true : (*(*this) == (*rhs)) //两者都未初始化,返回true
//两者都初始化时,比较两个T对象是否相等
);
} bool operator<(const Optional<T>& rhs) const
{
bool bInit = bool(*this);
return !rhs ? false : (!bInit ? true : (*(*this) < (*rhs)));
} bool operator!=(const Optional<T>& rhs) const
{
return !(*this == rhs);
} bool isInit() const {return m_hasInit;} ~Optional()
{
destroy();
} }; #endif

//main.cpp

#include "Optional.hpp"
#include <iostream> using namespace std; struct Test
{
Test() : m_a(), m_b(){}
Test(int a, int b) : m_a(a), m_b(b){} int m_a;
int m_b;
void show()
{
cout << "a = "<< m_a << ", b = " << m_b << endl;
}
}; void TestOptional()
{
const Optional<string> a("ok");
Optional<string> b("ok");
Optional<string> c("aa");
Optional<string> d = b;
Optional<string> e; cout << (e<b) << endl; //true
cout << (b==d) << endl; //true
cout << *c << endl;
//cout << *e << endl; //error Optional<Test> op;
op.emplace(, );
(*op).show(); Test t;
if(op) //判断op是否被初始化
t = *op;
t.show(); op.emplace(, );
t = *op;
t.show();
} int main()
{
TestOptional();
return ;
} /*输出结果:
e:\Study\C++11\23>g++ -std=c++11 test.cpp
e:\Study\C++11\23>a.exe
1
1
aa
a = 1, b = 2
a = 1, b = 2
a = 3, b = 4
*/

2. 惰性求值:Lazy类的实现

(1)Lazy类的功能

  ①惰性求值一般用于函数式编程语言中。

  ②可实现函数的延迟调用,函数参数被绑定后并不立即调用,而是在以后的某个时候调用。

  ③可实现大对象数据的延迟加载。如当初始化某个对象时,该对象引用了一个大对象,但很多时候并不马上获取该对象的数据,就可以延迟加载这个大对象。

(2)实现细节

  ①借助lambda表达式,将函数封装到lambda表达式中,而不是马上求值,在需要的时候再调用lambda表达式去求值

  ②std::function用于保存传入的函数,并延迟到后面需要使用值的时候才执行,函数的返回值放到一个Optional对象中。Optional对象是否被初始化,来判断大对象是否己加载。

  ③辅助函数lazy的作用是方便使用Lazy类, Lazy<T>中的T用来表示返回值类型大对象的类型这也是被封装的函数返回值类型,可利用std::result_of来获取该返回值类型。

【编程实验】Lazy类的实现

//Lazy.hpp

#ifndef _LAZY_H_
#define _LAZY_H_ #include "Optional.hpp"
#include <functional>
#include <type_traits>
#include <utility> //for std::forward template<typename T>
struct Lazy
{
private:
Optional<T> m_value;
std::function<T()> m_func;
public:
Lazy(){} //保存需要延迟执行的函数及其参数
template<typename Func, typename ...Args>
Lazy(Func&& f, Args&&... args)
{
m_func = [&f, &args...]{return f(args...);};
} //延迟执行,将结果放到Optional中缓存起来,下次不用重新计算就可以直接返回结果
T& value()
{
if(! m_value.isInit()){
m_value = m_func();
} return *m_value;
} bool isValueCreated() const
{
return m_value.isInit();
}
}; //辅助函数,简化Lazy的调用
template<class Func, typename... Args>
Lazy<typename std::result_of<Func(Args...)>::type> //返回值类型Lazy<T>
lazy(Func&& fun, Args&&... args)
{
using ret_type_t = typename std::result_of<Func(Args...)>::type;
return Lazy<ret_type_t>(std::forward<Func>(fun), std::forward<Args>(args)...);
} #endif // _LAZY_H_

//main.cpp

#include "Lazy.hpp"
#include <iostream>
#include <memory> //for std::shared_ptr using namespace std; struct BigObject
{
BigObject()
{
cout << "lazy load big object..." << endl;
}
}; struct Test
{
private:
Lazy<std::shared_ptr<BigObject>> m_obj;
public:
Test()
{
m_obj = lazy([]{return std::make_shared<BigObject>();});
} void load()
{
m_obj.value();
}
}; int Foo(int x)
{
return x * ;
} void TestLazy()
{
//带参数的普通函数
int a = ;
auto lazy1 = lazy(Foo, a);
cout << lazy1.value() << endl; //8 //不带参数的lambda表达式
Lazy<int> lazy2 = lazy([]{return ;});
cout << lazy2.value() << endl; //12 //带参的function
std::function<int(int)> f = [](int x){return x + ;};
auto lazy3 = lazy(f, a);
cout << lazy3.value() << endl; //7 //延迟加载大对象
Test t;
t.load(); //lazy load big object...
} int main()
{
TestLazy();
return ;
}

第23课 可变参数模板(4)_Optional和Lazy类的实现的更多相关文章

  1. 第27课 可变参数模板(8)_TupleHelper

    1. TupleHelper的主要功能 (1)打印:由于tuple中的元素是可变参数模板,外部并不知道内部到底是什么数据,有时调试时需要知道其具体值,希望能打印出tuple中所有的元素值. (2)根据 ...

  2. 第26课 可变参数模板(7)_any和variant类的实现

    1. any类的实现 (1)any类: ①是一个特殊的,只能容纳一个元素的容器,它可以擦除类型,可以将何任类型的值赋值给它. ②使用时,需要根据实际类型将any对象转换为实际的对象. (2)实现any ...

  3. 第25课 可变参数模板(6)_function_traits和ScopeGuard的实现

    1. function_traits (1)function_traits的作用:获取函数的实际类型.返回值类型.参数个数和具体类型等.它能获取所有函数语义类型信息.可以获取普通函数.函数指针.std ...

  4. 第24课 可变参数模板(5)_DllHelper和lambda链式调用

    1. dll帮助类 (1)dll的动态链接 ①传统的调用方式:先调用LoadLibrary来加载dll,再定义函数指针类型,接着调用GetProcAddress获取函数地址.然后通过函数指针调用函数, ...

  5. C++反射机制:可变参数模板实现C++反射

    1. 概要   本文描述一个通过C++可变参数模板实现C++反射机制的方法.该方法非常实用,在Nebula高性能网络框架中大量应用,实现了非常强大的动态加载动态创建功能.Nebula框架在Github ...

  6. C++ 0x 使用可变参数模板类 实现 C# 的委托机制

    #ifndef _ZTC_DELEGATE_H_ #define _ZTC_DELEGATE_H_ #include <vector> #include <functional> ...

  7. c++11 可变参数模板类

    c++11 可变参数模板类 #define _CRT_SECURE_NO_WARNINGS #include <iostream> #include <string> #inc ...

  8. c++11 可变参数模板函数

    c++11 可变参数模板函数 #define _CRT_SECURE_NO_WARNINGS #include <iostream> #include <string> #in ...

  9. C++反射机制:可变参数模板实现C++反射(使用C++11的新特性--可变模版参数,只根据类的名字(字符串)创建类的实例。在Nebula高性能网络框架中大量应用)

    1. 概要   本文描述一个通过C++可变参数模板实现C++反射机制的方法.该方法非常实用,在Nebula高性能网络框架中大量应用,实现了非常强大的动态加载动态创建功能.Nebula框架在码云的仓库地 ...

随机推荐

  1. Mac上svn报错解决方案

    具体的报错信息为:xcrun: error: invalid active developer path (/Library/Developer/CommandLineTools), missing ...

  2. SharePoint REST API - 使用REST接口对列表设置自定义权限

    博客地址:http://blog.csdn.net/FoxDave SharePoint网站.列表和列表项都属于SecurableObject类型.默认情况下,一个安全对象继承父级的权限.对一个对 ...

  3. IOS Block代码块的定义与使用

    代码块的本质是和其他的变量类似,不同的是,代码块存储的数据是一个函数体.使用代码块,你可以像调用其他标准函数一样的调用,可以传入参数,并得到返回值.     脱字符是代码块的语法标记.下图表示代码块的 ...

  4. tomcat升级 遇到的坑

    今天说说tomcat升级后出的问题 以前的版本是8.0.30的 因用安全漏洞 需要升级tomcat 为8.5.28的版本 升级后jvm的配置 等等都和一起一样,过了几天发现,我们的错误日志和处理影响转 ...

  5. vagrant package制作一个box镜像

    1.进入virtualbox安装目录,查看虚拟机的名称(第一列为虚拟机名称) # vboxmanage list vms 2. vagrant  package 打包命令 vagrant packag ...

  6. python 转义字符 html 爬虫

    用python的requests包 抓取某些网页时,返回的html中,一些字段含有一些 转义字符 \\\\\\\ 这些转义字符给我们后期处理带来一些麻烦, 比方说 运行js等 python用print ...

  7. pycharm 在线激活

    1.在pycharm过期页面选择 enter License 2.激活界面的License server输入:http://idea.liyang.io 然后点击激活  (确保电脑能上网,亲测有效)

  8. Codeblocks中文乱码解决方法

    odeblocks中文乱码解决方法: 特别提示:出现中文乱码情况才执行以下操作,未出现请勿随意修改!!!! 打开Codeblocks -> 设置 -> 编辑器: 然后点击 Encoding ...

  9. [JAVA]对象的别名问题

    对于JAVA的基本数据类型,a=b就是把b的内容复制给a.若接着又修改了a,对b是没有影响的. 但是在为对象“赋值”的时候,情况发生了变化.对一个对象进行操作时,我们真正操作的是对象的引用. 下面对两 ...

  10. Html 页面载入内容前,显示 loading 效果。

    Html 内容 loading部分: <div id="sys-loading" class=""><div class="spin ...