欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解C++句柄類

 更新時(shí)間:2018年06月12日 08:24:55   作者:lzm_cn  
本篇文章給大家詳細(xì)分析了C++句柄類的相關(guān)知識(shí)點(diǎn),對(duì)此有需要的朋友跟著學(xué)習(xí)參考下吧。

上一篇文件介紹了關(guān)于C++代理類的使用場(chǎng)景和實(shí)現(xiàn)方法,但是代理類存在一定的缺陷,就是每個(gè)代理類會(huì)創(chuàng)建一個(gè)新的對(duì)象,無(wú)法避免一些不必要的內(nèi)存拷貝,本篇文章引入句柄類,在保持代理類多態(tài)性的同時(shí),還可以避免進(jìn)行不不要的對(duì)象復(fù)制。

我們先來(lái)看一個(gè)簡(jiǎn)易的字符串封裝類:MyString,為了方便查看代碼,將函數(shù)的聲明和實(shí)現(xiàn)放到了一起。

class MyString
{
public:
 // 默認(rèn)構(gòu)造函數(shù)
 MyString()
 {
  std::cout << "MyString()" << std::endl;

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

 // const char*參數(shù)的構(gòu)造函數(shù)
 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);
  }
 }

 // 拷貝構(gòu)造函數(shù)
 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 風(fēng)格字符串
 const char* c_str()
 {
  return buf_;
 }

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

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

private:
 char* buf_;
 size_t len_;
};

看一段測(cè)試程序

#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;
}

輸出內(nèi)容如下:

可以看到,定義了三個(gè)MyString對(duì)象,str2和str3都是由str1拷貝構(gòu)造而來(lái),而且在程序的運(yùn)行過(guò)程中,str2和str3的內(nèi)容并未被修改,但是str1和str2已經(jīng)復(fù)制了str1緩沖區(qū)的內(nèi)容到自己的緩沖區(qū)中。其實(shí)這里可以做一個(gè)優(yōu)化,就是讓str1和str2在拷貝構(gòu)造的時(shí)候,直接指向str1的內(nèi)存,這樣就避免了重復(fù)的內(nèi)存拷貝。但是這樣又會(huì)引出一些新的問(wèn)題:

1. 多個(gè)指針指向同一塊動(dòng)態(tài)內(nèi)存,內(nèi)存改何時(shí)釋放?由誰(shuí)釋放?

2. 如果某個(gè)對(duì)象需要修改字符串中的內(nèi)容,該如和處理?

解決這些問(wèn)題,在C++中有兩個(gè)比較經(jīng)典的方案,那就是引用計(jì)數(shù)和Copy On Write。

在引用計(jì)數(shù)中,每一個(gè)對(duì)象負(fù)責(zé)維護(hù)對(duì)象所有引用的計(jì)數(shù)值。當(dāng)一個(gè)新的引用指向?qū)ο髸r(shí),引用計(jì)數(shù)器就遞增,當(dāng)去掉一個(gè)引用時(shí),引用計(jì)數(shù)就遞減。當(dāng)引用計(jì)數(shù)到零時(shí),該對(duì)象就將釋放占有的資源。

下面給出引用計(jì)數(shù)的一個(gè)封裝類:

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)
 {
  // 更新原引用計(jì)數(shù)的信息
  if (Only())
  {
   delete count_;
  }
  else
  {
   --*count_;
  }

  // 更新新的引用計(jì)數(shù)的信息
  ++*other.count_;
  
  // 綁定到新的引用計(jì)數(shù)
  count_ = other.count_;
 }

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

private:
 int* count_;
};

Copy On Write:就是寫(xiě)時(shí)復(fù)制,通過(guò)拷貝構(gòu)造初始化對(duì)象時(shí),并不直接將參數(shù)的資源往新的對(duì)象中復(fù)制一份,而是在需要修改這些資源時(shí),將原有資源拷貝過(guò)來(lái),再進(jìn)行修改,就避免了不必要的內(nèi)存拷貝。

下面的代碼是完整的句柄類MyStringHandle。每一個(gè)句柄類,都包含一個(gè)引用計(jì)數(shù)的類,用來(lái)管理和記錄對(duì)MyString對(duì)象的引用次數(shù)。

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

 // 這兩種參數(shù)的構(gòu)造函數(shù)必須構(gòu)造一個(gè)新的MyString對(duì)象出來(lái)
 MyStringHandle(const char* str) : pstr_(new MyString(str)) {}
 MyStringHandle(const MyString& other) : pstr_(new MyString(other)) {}

 // 拷貝構(gòu)造函數(shù),將指針綁定到參數(shù)綁定的對(duì)象上,引用計(jì)數(shù)直接拷貝構(gòu)造,在拷貝構(gòu)造函數(shù)內(nèi)更新引用計(jì)數(shù)的相關(guān)信息
 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)
 {
  // 綁定在同一個(gè)對(duì)象上的句柄相互賦值,不作處理
  if (other.pstr_ == pstr_)
  {
   return *this;
  }

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

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

  return *this;
 }

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

  return *this;
 }

private:
 MyString* pstr_;
 RefCount ref_count_;
};

看一段測(cè)試程序:

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

 // 不會(huì)構(gòu)造新的MyString
 MyStringHandle str2 = str1;
 MyStringHandle str3 = str1;
 MyStringHandle str4 = str1;

 // 構(gòu)造一個(gè)空的MyString
 MyStringHandle str5;

 // 將str1賦值到str5,不會(huì)有內(nèi)存拷貝
 str5 = str1;

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

 return 0;
}

總結(jié)

本篇文章介紹了C++句柄類的設(shè)計(jì)思想與簡(jiǎn)單實(shí)現(xiàn),主要通過(guò)引用計(jì)數(shù)和Copy On Write實(shí)現(xiàn),這兩種思想還是很經(jīng)典的,垃圾回收、智能指針的實(shí)現(xiàn)都有借鑒這兩種思想。水平有限,可能會(huì)有一些錯(cuò)誤或者描述不明確,歡迎大家拍磚~~

相關(guān)文章

  • 簡(jiǎn)單介紹C語(yǔ)言中的umask()函數(shù)和truncate()函數(shù)

    簡(jiǎn)單介紹C語(yǔ)言中的umask()函數(shù)和truncate()函數(shù)

    這篇文章主要介紹了簡(jiǎn)單介紹C語(yǔ)言中的umask()函數(shù)和truncate()函數(shù),是C語(yǔ)言入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下
    2015-09-09
  • QML中動(dòng)態(tài)與靜態(tài)模型應(yīng)用詳解

    QML中動(dòng)態(tài)與靜態(tài)模型應(yīng)用詳解

    QML是一種描述性的腳本語(yǔ)言,文件格式以.qml結(jié)尾。語(yǔ)法格式非常像CSS(參考后文具體例子),但又支持javascript形式的編程控制。QtDesigner可以設(shè)計(jì)出·ui界面文件,但是不支持和Qt原生C++代碼的交互
    2022-08-08
  • 基于Windows API實(shí)現(xiàn)遍歷所有文件并刪除的方法

    基于Windows API實(shí)現(xiàn)遍歷所有文件并刪除的方法

    這篇文章主要介紹了基于Windows API實(shí)現(xiàn)遍歷所有文件并刪除的方法,是win32應(yīng)用程序的一個(gè)比較典型的文件操作應(yīng)用技巧,需要的朋友可以參考下
    2015-04-04
  • VSCode配置C/C++并添加非工作區(qū)頭文件的方法

    VSCode配置C/C++并添加非工作區(qū)頭文件的方法

    這篇文章主要介紹了VSCode配置C/C++并添加非工作區(qū)頭文件的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-03-03
  • C++實(shí)現(xiàn)LeetCode(88.混合插入有序數(shù)組)

    C++實(shí)現(xiàn)LeetCode(88.混合插入有序數(shù)組)

    這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(88.混合插入有序數(shù)組),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • C++中的auto_ptr智能指針的作用及使用方法詳解

    C++中的auto_ptr智能指針的作用及使用方法詳解

    這篇文章主要介紹了C++中的auto_ptr智能指針的作用及使用方法詳解的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2016-07-07
  • C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)深入探索順序表

    C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)深入探索順序表

    順序表,全名順序存儲(chǔ)結(jié)構(gòu),是線性表的一種,線性表用于存儲(chǔ)邏輯關(guān)系為“一對(duì)一”的數(shù)據(jù),順序表自然也不例外,不僅如此,順序表對(duì)數(shù)據(jù)的物理存儲(chǔ)結(jié)構(gòu)也有要求,跟隨下文來(lái)具體了解吧
    2022-03-03
  • C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單飛機(jī)大戰(zhàn)

    C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單飛機(jī)大戰(zhàn)

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單飛機(jī)大戰(zhàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-02-02
  • C語(yǔ)言按關(guān)鍵字搜索文件夾中文件的方法

    C語(yǔ)言按關(guān)鍵字搜索文件夾中文件的方法

    這篇文章主要介紹了C語(yǔ)言按關(guān)鍵字搜索文件夾中文件的方法,涉及C語(yǔ)言文件操作及字符串查找的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-07-07
  • c++ 構(gòu)造函數(shù)的初始化列表

    c++ 構(gòu)造函數(shù)的初始化列表

    構(gòu)造函數(shù)的初始化列表僅僅指定用于初始化成員的值,并不指定這些初始化執(zhí)行的次序。成員初始化的次序就是定義成員的次序,第一個(gè)成員首先被初始化,然后是第二個(gè),依次類推
    2013-07-07

最新評(píng)論