Sting类的拷贝构造

在自己定义的String类中使用写实拷贝来提高效率。

写实拷贝是在浅拷贝的基础上使用引用计数器来拷贝对象,如果有对象被拷贝构造时只需要将计数器++,不用再开辟新的空间。

初版的代码:

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
class String
{
public:
    String(const char* str = "")
        :_str(new char[strlen(str) + 1])
        ,_refcount(1)
    {
        strcpy(_str, str);
       //每当有新的对象构造时,将计数器置1,意思是有一个对象引用这块空间
    }
    ~String()
    {
        if (--_refcount == 0) //如果只有一个对象引用这块空间,则直接释放,否则计数器--
        {
            delete[]_str;
        }
    }
    //写实拷贝
    String(String& s)
        :_str(s._str)
    {
        _refcount++;
    }
private:
    char* _str;
    int _refcount;
};
 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30

因为计数器是私有的,所以每个对象的计数器都为1,释放空间时系统会奔溃。 
我们还可以将_refcount设置为static静态,为对象共有。但是static是全局变量,所以创建不同对象时_refcount会出现问题。

所以可以直接再开辟一个空间用来存储计数器个数 ,使用一个指针指向这块空间,这样拷贝构造时只需要将对象指针指向计数器,将计数器内容++,释放对象空间时,如果计数器个数不为1,则直接计数器内容–,为1证明只有一个对象指向这块空间,则释放计数器空间,释放字符串空间。 

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
class String
{
public:
    String(const char* str = "")
        :_str(new char[strlen(str) + 1])
        , _refptr(new int(1))
    {
        strcpy(_str, str);
    }
    ~String()
    {
        if (--(*_refptr) == 0)
        {
            delete[]_str;
            delete _refptr;
        }
    }
    String(String& s)
        :_str(s._str)
        , _refptr(s._refptr)
    {
        (*_refptr)++;
    }
    String& operator=(String& s)
    {
        //自己给自己赋值,或者两个指向同一块空间
        //自己拥有一块空间
        //自己和别人共同拥有一块空间
        if (_str != s._str)
        {
            if (--(*_refptr) == 0)
            {
                delete[]_str;
                delete _refptr;
            }
            _str = s._str;
            _refptr = s._refptr;
            (*_refptr)++;
        }
        return *this;
    }
private:
    char* _str;
    int* _refptr;
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值