# STL设计之EBO(空基类优化) ## 0.导语 EBO简称Empty Base Optimization。 本节从空类开始,到STL内部,到测试,再到我们自己实现一个EBO,对比性能,最后再测试,总结。 ## 1.空类 定义一个空类:没有成员变量,没有继承,没有数据元素的类。 ```cpp class Empty{ public: void print() { std::cout<<"I am Empty class"< bool isSame( T const & t1, T const & t2 ) { return &t1 == &t2; } ``` 我们来测试一下: ```cpp int main() { Empty a,b; assert(!isSame(a,b)); // 编译通过,a与b的地址不同 Empty *p=new Empty; Empty *q=new Empty; assert(!isSame(p,q)); // 编译通过,a与b的地址不同 return 0; } ``` 上面测试了,两个不同对象地址是不同的,考虑下面场景: ```cpp Empty a,b; // 在同一地址具有两个对象将意味着在使用指针引用它们时将无法区分这两个对象。 Empty *p1=&a; p1->print(); ``` 此时会发现,如果a的地址与b的地址一样,那么在同一地址具有两个对象将意味着在使用指针引用它们时将无法区分这两个对象。因此两个不同对象的地址不同。 ## 2.空基类优化 现在对比一下下面两个用法,第一种,一个类中包含了两一个类作为成员,然后通过这个来获得被包含类的功能。 ```cpp class notEbo { int i; Empty e; // do other things }; ``` 另一种直接采用继承的方式来获得基类的成员函数及其他功能等等。 ```cpp class ebo:public Empty { int i; // do other things }; ``` 接下来做个测试: ```cpp std::cout< __gnu_cxx::bitmap_allocator<_Tp> __gnu_cxx::bitmap_allocator<_Tp> __gnu_cxx::__mt_alloc<_Tp> __gnu_cxx::__pool_alloc<_Tp> __gnu_cxx::malloc_allocator<_Tp> ``` 那这和我们的EBO有啥关系呢? 实际上,上面所列出继承的基类都是内存管理的EBO(空基类)。 在每个容器中的使用都是调用每个内存管理的`rebind<_Tp>::other`。 例如红黑树源码结构: ```cpp typedef typename __gnu_cxx::__alloc_traits<_Alloc>::template rebind<_Rb_tree_node<_Val> >::other _Node_allocator; struct _Rb_tree_impl : public _Node_allocator { // do somethings }; ``` 接下来我们看上面列出的内存管理类里面的源码结构:这里拿`allocator`举例: ```cpp template class allocator: public __allocator_base<_Tp> { template struct rebind { typedef allocator<_Tp1> other; }; }; ``` 看到了没,通过`rebind<_Tp>::other`来获得传递进来的内存分配器,也就是前面提到的这些。 ```cpp std::allocator<_Tp> __gnu_cxx::bitmap_allocator<_Tp> __gnu_cxx::bitmap_allocator<_Tp> __gnu_cxx::__mt_alloc<_Tp> __gnu_cxx::__pool_alloc<_Tp> __gnu_cxx::malloc_allocator<_Tp> ``` 搞懂了这些,来测试一波: ```cpp void print() { cout<)<<" "<::rebind::other)<)<<" "<::rebind::other)<)<<" "<::rebind::other)<)<<" "<::rebind::other)<)<<" "<::rebind::other)<)<<" "<::rebind::other)< class MyContainerNotEBO { T *data_ = nullptr; std::size_t capacity_; Allocator allocator_; // 嵌入一个MyAllocator public: MyContainerNotEBO(std::size_t capacity) : capacity_(capacity), allocator_(), data_(nullptr) { std::cout << "alloc malloc" << std::endl; data_ = reinterpret_cast(allocator_.allocate(capacity * sizeof(T))); // 分配内存 } ~MyContainerNotEBO() { std::cout << "MyContainerNotEBO free malloc" << std::endl; allocator_.deallocate(data_); } }; ``` 第二种方式:采用空基类优化,继承来获得内存管理功能 ```cpp template class MyContainerEBO : public Allocator { // 继承一个EBO T *data_ = nullptr; std::size_t capacity_; public: MyContainerEBO(std::size_t capacity) : capacity_(capacity), data_(nullptr) { std::cout << "alloc malloc" << std::endl; data_ = reinterpret_cast(this->allocate(capacity * sizeof(T))); } ~MyContainerEBO() { std::cout << "MyContainerEBO free malloc" << std::endl; this->deallocate(data_); } }; ``` 开始测试: ```cpp int main() { MyContainerNotEBO notEbo = MyContainerNotEBO(0); std::cout << "Using Not EBO Test sizeof is " << sizeof(notEbo) << std::endl; MyContainerEBO ebo = MyContainerEBO(0); std::cout << "Using EBO Test sizeof is " << sizeof(ebo) << std::endl; return 0; } ``` 测试结果: ```cpp alloc malloc Using Not EBO Test sizeof is 24 alloc malloc Using EBO Test sizeof is 16 MyContainerEBO free malloc MyContainerNotEBO free malloc ``` 我们发现采用EBO的设计确实比嵌入设计好很多。至此,本节学习完毕。