中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

C++句柄類怎么應用

發布時間:2022-04-14 17:18:48 來源:億速云 閱讀:188 作者:zzz 欄目:編程語言

這篇文章主要介紹了C++句柄類怎么應用的相關知識,內容詳細易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇C++句柄類怎么應用文章都會有所收獲,下面我們一起來看看吧。

我們先來看一個簡易的字符串封裝類:MyString,為了方便查看代碼,將函數的聲明和實現放到了一起。

class MyString
{
public:
 // 默認構造函數
 MyString()
 {
  std::cout << "MyString()" << std::endl;

  buf_ = new char[1];
  buf_[0] = '\0';
  len_ = 0;
 }

 // const char*參數的構造函數
 MyString(const char* str)
 {
  std::cout << "MyString(const char* str)" << std::endl;

  if (str == nullptr)
  {
   len_ = 0;
   buf_ = new char[1];
   buf_[0] = '\0';
  }
  else
  {
   len_ = strlen(str);
   buf_ = new char[len_ + 1];
   strcpy_s(buf_, len_ + 1, str);
  }
 }

 // 拷貝構造函數
 MyString(const MyString& other)
 {
  std::cout << "MyString(const MyString& other)" << std::endl;

  len_ = strlen(other.buf_);
  buf_ = new char[len_ + 1];
  strcpy_s(buf_, len_ + 1, other.buf_);
 }

 // str1 = str2;
 const MyString& operator=(const MyString& other)
 {
  std::cout << "MyString::operator=(const MyString& other)" << std::endl;

  // 判斷是否為自我賦值
  if (this != &other)
  {
   if (other.len_ > this->len_)
   {
    delete[]buf_;
    buf_ = new char[other.len_ + 1];
   }

   len_ = other.len_;
   strcpy_s(buf_, len_ + 1, other.buf_);
  }

  return *this;
 }

 // str = "hello!";
 const MyString& operator=(const char* str)
 {
  assert(str != nullptr);

  std::cout << "operator=(const char* str)" << std::endl;

  size_t strLen = strlen(str);
  if (strLen > len_)
  {
   delete[]buf_;
   buf_ = new char[strLen + 1];
  }

  len_ = strLen;
  strcpy_s(buf_, len_ + 1, str);
  
  return *this;
 }
 
 // str += "hello"
 void operator+=(const char* str)
 {
  assert(str != nullptr);

  std::cout << "operator+=(const char* str)" << std::endl;

  if (strlen(str) == 0)
  {
   return;
  }

  size_t newBufLen = strlen(str) + len_ + 1;
  char* newBuf = new char[newBufLen];
  strcpy_s(newBuf, newBufLen, buf_);
  strcat_s(newBuf, newBufLen, str);

  delete[]buf_;
  buf_ = newBuf;

  len_ = strlen(buf_);
 }

 // 重載 ostream的 <<操作符 ,支持 std::cout << MyString 的輸出
 friend std::ostream& operator<<(std::ostream &out, MyString& obj)
 {
  out << obj.c_str();
  return out;
 }

 // 返回 C 風格字符串
 const char* c_str()
 {
  return buf_;
 }

 // 返回字符串長度
 size_t length()
 {
  return len_;
 }

 ~MyString()
 {
  delete[]buf_;
  buf_ = nullptr;
 }

private:
 char* buf_;
 size_t len_;
};

看一段測試程序

#include "MyString.h"

int _tmain(int argc, _TCHAR* argv[])
{
 MyString str1("hello~~");
 MyString str2 = str1;
 MyString str3 = str1;

 std::cout << "str1=" << str1 << ", str2=" << str2 << ", str3=" << str3;

 return 0;
}

可以看到,定義了三個MyString對象,str2和str3都是由str1拷貝構造而來,而且在程序的運行過程中,str2和str3的內容并未被修改,但是str1和str2已經復制了str1緩沖區的內容到自己的緩沖區中。其實這里可以做一個優化,就是讓str1和str2在拷貝構造的時候,直接指向str1的內存,這樣就避免了重復的內存拷貝。但是這樣又會引出一些新的問題:

1. 多個指針指向同一塊動態內存,內存改何時釋放?由誰釋放?

2. 如果某個對象需要修改字符串中的內容,該如和處理?

解決這些問題,在C++中有兩個比較經典的方案,那就是引用計數和Copy On Write。

在引用計數中,每一個對象負責維護對象所有引用的計數值。當一個新的引用指向對象時,引用計數器就遞增,當去掉一個引用時,引用計數就遞減。當引用計數到零時,該對象就將釋放占有的資源。

下面給出引用計數的一個封裝類:

class RefCount
{
public:

 RefCount() : count_(new int(1)){};

 RefCount(const RefCount& other) : count_(other.count_)
 {
  ++*count_;
 }

 ~RefCount()
 {
  if (--*count_ == 0)
  {
   delete count_;
   count_ = nullptr;
  }
 }

 bool Only()
 {
  return *count_ == 1;
 }

 void ReAttach(const RefCount& other)
 {
  // 更新原引用計數的信息
  if (Only())
  {
   delete count_;
  }
  else
  {
   --*count_;
  }

  // 更新新的引用計數的信息
  ++*other.count_;
  
  // 綁定到新的引用計數
  count_ = other.count_;
 }

 void MakeNewRef()
 {
  if (*count_ > 1)
  {
   --*count_;
   count_ = new int(1);
  }
 }

private:
 int* count_;
};

Copy On Write:就是寫時復制,通過拷貝構造初始化對象時,并不直接將參數的資源往新的對象中復制一份,而是在需要修改這些資源時,將原有資源拷貝過來,再進行修改,就避免了不必要的內存拷貝。

下面的代碼是完整的句柄類MyStringHandle。每一個句柄類,都包含一個引用計數的類,用來管理和記錄對MyString對象的引用次數。

class MyStringHandle
{
public:
 MyStringHandle() : pstr_(new MyString){}

 // 這兩種參數的構造函數必須構造一個新的MyString對象出來
 MyStringHandle(const char* str) : pstr_(new MyString(str)) {}
 MyStringHandle(const MyString& other) : pstr_(new MyString(other)) {}

 // 拷貝構造函數,將指針綁定到參數綁定的對象上,引用計數直接拷貝構造,在拷貝構造函數內更新引用計數的相關信息
 MyStringHandle(const MyStringHandle& ohter) : ref_count_(ohter.ref_count_), pstr_(ohter.pstr_) {}

 ~MyStringHandle()
 {
  if (ref_count_.Only())
  {
   delete pstr_;
   pstr_ = nullptr;
  }
 }

 MyStringHandle& operator=(const MyStringHandle& other)
 {
  // 綁定在同一個對象上的句柄相互賦值,不作處理
  if (other.pstr_ == pstr_)
  {
   return *this;
  }

  // 若當前引用唯一,則銷毀當前引用的MyString
  if (ref_count_.Only())
  {
   delete pstr_;
  }

  // 分別將引用計數和對象指針重定向
  ref_count_.ReAttach(other.ref_count_);
  pstr_ = other.pstr_;

  return *this;
 }

 // str = "abc" 這里涉及到對字符串內容的修改,
 MyStringHandle& operator=(const char* str)
 {
  if (ref_count_.Only())
  {
   // 如果當前句柄對MyString對象為唯一的引用,則直接操作改對象進行賦值操作
   *pstr_ = str;
  }
  else
  {
   // 如果不是唯一引用,則將原引用數量-1,創建一個新的引用,并且構造一個新的MyString對象
   ref_count_.MakeNewRef();
   pstr_ = new MyString(str);
  }

  return *this;
 }

private:
 MyString* pstr_;
 RefCount ref_count_;
};

看一段測試程序:

int _tmain(int argc, _TCHAR* argv[])
{
 // 構造MyString
 MyStringHandle str1("hello~~");

 // 不會構造新的MyString
 MyStringHandle str2 = str1;
 MyStringHandle str3 = str1;
 MyStringHandle str4 = str1;

 // 構造一個空的MyString
 MyStringHandle str5;

 // 將str1賦值到str5,不會有內存拷貝
 str5 = str1;

 // 修改str5的值
 str5 = "123";
 str5 = "456";

 return 0;
}

關于“C++句柄類怎么應用”這篇文章的內容就介紹到這里,感謝各位的閱讀!相信大家對“C++句柄類怎么應用”知識都有一定的了解,大家如果還想學習更多知識,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

c++
AI

安新县| 安徽省| 龙川县| 金华市| 马关县| 白城市| 黄龙县| 伊吾县| 澄迈县| 汽车| 宝鸡市| 德昌县| 通渭县| 永川市| 松阳县| 怀柔区| 贞丰县| 馆陶县| 梓潼县| 清徐县| 墨江| 莱芜市| 达孜县| 阳原县| 瑞安市| 三江| 容城县| 仙游县| 蓝山县| 喀什市| 宁强县| 玉树县| 临沧市| 年辖:市辖区| 青龙| 台中县| 武山县| 福贡县| 菏泽市| 岳西县| 北川|