成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

深度剖析 C++ 對象池自動回收技術實現

開發 后端
對象池可以顯著提高性能,如果一個對象的創建非常耗時或非常昂貴,頻繁去創建的話會非常低效。對象池通過對象復用的方式來避免重復創建對象,它會事 先創建一定數量的對象放到池中,當用戶需要創建對象的時候,直接從對象池中獲取即可,用完對象之后再放回到對象池中,以便復用。

對象池可以顯著提高性能,如果一個對象的創建非常耗時或非常昂貴,頻繁去創建的話會非常低效。對象池通過對象復用的方式來避免重復創建對象,它會事 先創建一定數量的對象放到池中,當用戶需要創建對象的時候,直接從對象池中獲取即可,用完對象之后再放回到對象池中,以便復用。這種方式避免了重復創建耗 時或耗資源的大對象,大幅提高了程序性能。本文將探討對象池的技術特性以及源碼實現。

對象池類圖

  • ObjectPool:管理對象實例的pool。

  • Client:使用者。

適用性:

  • 類的實例可重用。

  • 類的實例化過程開銷較大。

  • 類的實例化的頻率較高。

效果:

  • 節省了創建類實例的開銷。

  • 節省了創建類實例的時間。

  • 存儲空間隨著對象的增多而增大。

問題

目前縱觀主流語言的實現方式無外乎3個步驟:

  1. 初始創建一定數量的對象池(也允許從外面添加對象)。

  2. 從對象池中取對象來使用。

  3. 用完之后返回對象池。

一般情況下這樣是OK的,可能存在的問題是在第三步,有兩個問題:

  1. 不方便,每次都需要顯式回收對象。

  2. 忘記將對象放回對象池,造成資源浪費。

改進動機

解決顯式回收的問題,實現自動回收,省心省力。改進之后的對象池無須提供release方法,對象會自動回收,改進之后的類圖如下。


技術內幕

借助c++11智能指針,因為智能指針可以自定義刪除器,在智能指針釋放的時候會調用刪除器,在刪除器中我們將用完的對象重新放回對象池。思路比較簡單,但實現的時候需要考慮兩個問題:

  1. 什么時候定義刪除器?

  2. 用shared_ptr還是unique_ptr?

1. 什么時候定義刪除器

自定義刪除器只做一件事,就是將對象重新放入對象池。如果對象池初始化的時候就自定義刪除器的話,刪除器中的邏輯是將對象放回對象池,放回的時候無 法再定義一個這樣的刪除器,所以這種做法行不通。需要注意,回收的對象只能是默認刪除器的。除了前述原因之外,另外一個原因是對象池釋放的時候需要釋放所 有的智能指針,釋放的時候如果存在自定義刪除器將會導致對象無法刪除。只有在get的時候定義刪除器才行,但是初始創建或加入的智能指針是默認刪除器,所 以我們需要把智能指針的默認刪除器改為自定義刪除器。

1.2 用shared_ptr還是unique_ptr

因為我們需要把智能指針的默認刪除器改為自定義刪除器,用shared_ptr會很不方便,因為你無法直接將shared_ptr的刪除器修改為自 定義刪除器,雖然你可以通過重新創建一個新對象,把原對象拷貝過來的做法來實現,但是這樣做效率比較低。而unique_ptr由于是獨占語義,提供了一 種簡便的方法方法可以實現修改刪除器,所以用unique_ptr是最適合的。

1.3 實現源碼

 

  1. #pragma once 
  2. #include <memory> 
  3. #include <vector> 
  4. #include <functional> 
  5.  
  6. template <class T> 
  7. class SimpleObjectPool 
  8. public
  9.     using DeleterType = std::function<void(T*)>; 
  10.  
  11.     void add(std::unique_ptr<T> t) 
  12.     { 
  13.         pool_.push_back(std::move(t)); 
  14.     } 
  15.  
  16.     std::unique_ptr<T, DeleterType> get() 
  17.     { 
  18.         if (pool_.empty()) 
  19.         { 
  20.             throw std::logic_error("no more object"); 
  21.         } 
  22.  
  23.         //every time add custom deleter for default unique_ptr 
  24.         std::unique_ptr<T, DeleterType> ptr(pool_.back().release(), [this](T* t) 
  25.         { 
  26.             pool_.push_back(std::unique_ptr<T>(t)); 
  27.         }); 
  28.  
  29.         pool_.pop_back(); 
  30.         return std::move(ptr); 
  31.     } 
  32.  
  33.     bool empty() const 
  34.     { 
  35.         return pool_.empty(); 
  36.     } 
  37.  
  38.     size_t size() const 
  39.     { 
  40.         return pool_.size(); 
  41.     } 
  42.  
  43. private
  44.     std::vector<std::unique_ptr<T>> pool_; 
  45. }; 
  46.  
  47. //test code 
  48. void test_object_pool() 
  49.     SimpleObjectPool<A> p; 
  50.     p.add(std::unique_ptr<A>(new A())); 
  51.     p.add(std::unique_ptr<A>(new A())); 
  52.     { 
  53.         auto t = p.get(); 
  54.         p.get(); 
  55.     } 
  56.  
  57.     { 
  58.         p.get(); 
  59.         p.get(); 
  60.     } 
  61.  
  62.     std::cout << p.size() << std::endl; 
  63. } 

如果你堅持用shared_ptr,那么回收的時候你需要這樣寫:

  1. std::shared_ptr<T> get() 
  2. if (pool_.empty()) 
  3.   throw std::logic_error("no more object"); 
  4.  
  5. std::shared_ptr<T> ptr = pool_.back(); 
  6. auto p = std::shared_ptr<T>(new T(std::move(*ptr.get())), [this](T* t) 
  7.   pool_.push_back(std::shared_ptr<T>(t)); 
  8. }); 
  9.  
  10. //std::unique_ptr<T, DeleterType> ptr(pool_.back().release(), [this](T* t) 
  11. //{ 
  12. // pool_.push_back(std::unique_ptr<T>(t)); 
  13. //}); 
  14.  
  15. pool_.pop_back(); 
  16. return p; 

這種方式需要每次都創建一個新對象,并且拷貝原來的對象,是一種比較低效的做法。代碼僅僅是為了展示如何實現自動回收對象,沒有考慮線程安全、對象池擴容策略等細節,源碼鏈接:object_pool

總結凡是需要自動回收的場景下都可以使用這種方式:在獲取對象的時候將默認刪除器改為自定義刪除器,確保它可以回收。注意,回收的智能指針使用的是 默認刪除器,可以確保對象池釋放時能正常釋放對象。同時也將獲取對象和釋放對象時,對象的控制權完全分離。其他的一些應用場景:多例模式,無需手動釋放, 自動回收。

責任編輯:王雪燕 來源: 祁宇
相關推薦

2021-08-29 22:05:04

對象自動回收

2010-01-28 16:31:54

C++類型

2010-01-15 10:32:21

C++語言

2010-01-26 14:46:42

C++語言

2010-01-13 13:42:55

C++編譯器

2010-01-26 17:16:33

C++應用程序

2010-01-28 14:54:01

C++資源管理

2010-01-13 10:16:42

C++軟件

2010-01-27 15:50:23

C++復雜性

2012-02-17 10:50:10

Java

2023-02-21 15:26:26

自動駕駛特斯拉

2010-01-11 10:19:57

C++開發工具

2010-01-11 17:43:23

C++程序設計

2010-01-13 11:14:06

C++虛表

2010-01-12 15:03:33

C++代碼

2009-09-10 17:37:01

C# get post

2010-01-14 17:42:47

CC++

2010-02-03 16:56:24

Python包

2025-05-06 09:12:46

2010-02-02 13:22:06

Python面向對象
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品一区一区 | 中文字幕亚洲欧美 | 天天综合网天天综合色 | 岛国在线免费观看 | 91操操操 | 日日操夜夜操天天操 | 毛片在线看片 | 日韩超碰 | 精产国产伦理一二三区 | 国产一区二区三区 | 国产精品久久久久久久免费大片 | 欧美日韩1区2区3区 欧美久久一区 | 中文字幕亚洲区一区二 | 午夜av电影 | 精品亚洲一区二区 | 久久久久久久久久影视 | 91精品国产综合久久久久 | 国产二区av | 99亚洲 | 日韩国产一区二区三区 | 国产精品久久久久久久久久久久 | 免费电影av | 国产视频一区二区 | 欧美日韩亚洲三区 | 中文字幕亚洲视频 | 久久精品综合网 | 亚洲综合在线视频 | www国产成人 | 久久久久国产一区二区三区不卡 | 国产高清在线 | 亚洲国产区 | 精品国产一区二区在线 | 国产精品永久 | 日本成人免费网站 | 亚洲一区二区日韩 | 91综合在线视频 | 亚洲av毛片成人精品 | 91文字幕巨乱亚洲香蕉 | 亚洲免费一 | 日韩在线精品强乱中文字幕 | 国产精品免费一区二区三区四区 |