使用引用计数和copy-on_write实现String类

时间:2023-02-20 18:11:45

这算是我开始复习的内容吧,关于 String 类半年前写过,最近拿出来溜溜,免得面试被问到而自己又忘记了。

之前的文章地址:[C++ 引用计数思想--利用引用计数器自定义 String 类](C++ 引用计数思想--利用引用计数器自定义 String 类)。

首先上一个 String 类最简明的写法,没有用到引用计数和 COW,不过写法实在是很简单,不容易出错。先看代码,然后说弊端。

#include <iostream>
#include <string.h> class my_string {
public:
my_string(const char* str = NULL) {
if(str == NULL){
str_ = new char[1];
*str_ = '\0';
}
else{
str_ = new char[strlen(str)+1];
strcpy(str_, str);
}
}
my_string(const my_string& other)
: str_(new char[other.size()]+1) { //直接使用参数列表
strcpy(str_, other.c_str());
}
my_string& operator=(my_string other) { //按值传递
swap(other);
return *this;
}
~my_string() {
//delete []str_;
str_ = NULL;
}
public:
size_t size() const {
return strlen(str_);
}
const char* c_str() const {
return str_;
}
void swap(my_string& other) {
std::swap(str_, other.str_);
}
public:
void show() const {
std::cout<<str_<<std::endl;
}
private:
char* str_;
};

上述就是最简洁方案的代码。不过,这种方案有一个弊端,甚至是错误。那就是我们不能在析构函数中直接 delete []str_ 。因为上述方案,可能造成两个对象对字符串内存资源的共享。如果析构函数中直接 delete 掉内存,那么两个对象,意味着该内存要被 delete 两次。其结果可想而知。

为了解决这个问题,我们引入了引用计数思想。使用引用计数,对拥有字符串资源的对象数目进行计数。只有当拥有该字符串资源的对象数目为 0 时,才销毁字符串内存资源。这就能够保证该内存只被 delete 一次。

#include <string.h>
#include <iostream> class string_rep {
friend class my_string;
public:
string_rep(const char* str = NULL) : use_count_(1) {
if(str == NULL){
str_ = new char[1];
*str_ = '\0';
}
else{
str_ = new char [strlen(str)+1];
strcpy(str_, str);
}
} //trivial copy assignment ~string_rep() {
delete []str_;
str_ = NULL;
}
public:
unsigned int use_count() const {
return use_count_;
}
const char* c_str() const {
return str_;
}
private:
void increment() {
++use_count_;
}
void decrement() {
if(--use_count_ == 0)
delete this;
}
private:
char *str_;
unsigned int use_count_;
}; class my_string {
public:
my_string(const char* str = NULL)
: rep(new string_rep(str)) {
}
my_string(const my_string& other) {
rep = other.rep;
rep->increment();
}
my_string& operator=(const my_string& other) {
if(this != &other){
rep->decrement(); //先减一
rep = other.rep;
rep->increment();
}
return *this;
}
~my_string() {
rep->decrement(); //不要忘记这步
}
public:
unsigned int use_count() const {
return rep->use_count();
}
const char* c_str() const {
return rep->c_str();
}
void tupper() {
if(use_count() > 1){
string_rep* new_rep = new string_rep(rep->str_);
rep->decrement();
rep = new_rep; //替换操作
} for(char *s=rep->str_; *s!='\0'; ++s)
*s -= 32;
}
private:
string_rep *rep;
};

解决了资源的销毁问题,那么还有一个新的问题:多个对象共享一个字符串资源,那么如果某个对象需要修改字符串,怎么处理?答案:COW 技术。

COW(copy on write) 技术,就是只有在写资源的时候才拷贝,读资源并不拷贝。修改字符串就属于写资源。例如上面的 tupper() 函数,这就是 copy-on-write 技术在string 类的一个简单实现。修改字符串时,我们直接 new 出一个新的 string_rep 类替换旧的,在新的 string_rep 上操作即可。不过要注意,要保证旧有资源的释放,否则会造成内存泄漏问题。

在上面的实现中,还有一个技巧,那就是 delete this。这可是一个争议的东西,不过在引用计数中,使用 delete this 是安全的的。因为引用计数中,就本例来说,string_rep 类仅暴露给了 my_string 类,而 delete_this 又是引用计数为 0 导致的,要么是在 my_string 类的析构函数中引起,要么是在 copy-on-write 函数中引起,这都是安全的。this 指针在 delete 之后并不会暴露给外部。

使用 delete this 的注意事项:

1.this 对象必须是用 new 操作符分配的(而不是 new[],也不是 placement new)。

2.dlete this 后,不能访问该对象的任何成员变量及虚函数。因为 delete this 会销毁成员原谅以及 vptr。但是注意,并不销毁 vtbl,因为 vtbl 是该类所有对象共有的,如果你知道 C++ 对象模型这就很好理解了,这里不赘述。

3.delete this 后,不能再访问 this 指针。

好了,以上这些就是关于string类的一些技巧,不过实际上有很优秀的编程思想蕴含在里面,慢慢提高吧。

使用引用计数和copy-on_write实现String类的更多相关文章

  1. Objective-C内存管理之-引用计数

    本文会继续深入学习OC内存管理,内容主要参考iOS高级编程,Objective-C基础教程,疯狂iOS讲义,是我学习内存管理的笔记 内存管理 1 内存管理的基本概念 1.1 Objective-C中的 ...

  2. 【转载】C&plus;&plus;应用引用计数技术

    原帖:http://www.cnblogs.com/chain2012/archive/2010/11/12/1875578.html 因为Windows的内核对象也运用了引用计数,所以稍作了解并非无 ...

  3. com关于引用计数

    实现引用计数并不难,但在什么层次上进行引用计数呢? 依照com规范,一个com组件能够实现多个com对象.而且每一个com对象又能够支持多个com接口,这样的层次结构为我们实现引用计数提供了多种选择方 ...

  4. 【python测试开发栈】python内存管理机制(一)—引用计数

    什么是内存 在开始进入正题之前,我们先来回忆下,计算机基础原理的知识,为什么需要内存.我们都知道计算机的CPU相当于人类的大脑,其运算速度非常的快,而我们平时写的数据,比如:文档.代码等都是存储在磁盘 ...

  5. String 类的实现(3)引用计数实现String类

    我们知道在C++中动态开辟空间时是用字符new和delete的.其中使用new test[N]方式开辟空间时实际上是开辟了(N*sizeof(test)+4)字节的空间.如图示其中保存N的值主要用于析 ...

  6. &lpar;转&rpar;C&plus;&plus;——std&colon;&colon;string类的引用计数

    1.概念 Scott Meyers在<More Effective C++>中举了个例子,不知你是否还记得?在你还在上学的时候,你的父母要你不要看电视,而去复习功课,于是你把自己关在房间里 ...

  7. String 类的实现(2)引用计数与写时拷贝

    1.引用计数 我们知道在C++中动态开辟空间时是用字符new和delete的.其中使用new test[N]方式开辟空间时实际上是开辟了(N*sizeof(test)+4)字节的空间.如图示其中保存N ...

  8. 从sizeof&lpar;string&rpar;到引用计数的漫游

    前言: 说是漫游,其实就是扯,一点一点的扯. 话说之前参加华为的德州扑克比赛,我用C++解析消息的时候碰到一个小问题,就是定长收消息的时候出错,在Linux下调了很久很久,终于发现,sizeof(st ...

  9. 【M29】引用计数

    1.引用计数这项技术,是为了让等值对象对象共享同一实体.此技术的发展有两个动机:a.记录堆上分配的对象,是垃圾回收机制的简单原理:b.节省内存,多个对象具有相同的值,存储多次很笨.速度更快,等值对象避 ...

随机推荐

  1. android Broadcast广播消息代码实现

    我用的是Fragment , 发送写在一个类中,接收写在另外一个类的内部类中.代码动态实现注册. 代码: myReceiver = new zcd.netanything.MyCar.myReceiv ...

  2. Python 2&period;7&lowbar;pandas连接MySQL数据处理&lowbar;20161229

    在我本地Mysql_local_db数据库建立了一个pandas数据表用来对pandas模块的学习 学习过程借鉴学习蓝鲸的网站分析笔记 1.创建表 CREATE TABLE pandastest( 城 ...

  3. MySQL基础操作命令

    MySQL基础操作命令 1. 查看MySQL进程 ps -ef|grep mysql |grep -v grep 2. 查看MySQL端口 ss -lnt | grep 3306 3. MySQL的启 ...

  4. Junit4学习笔记--方法的执行顺序

    package com.lt.Demo.TestDemo; import java.util.Arrays; import java.util.Collection; import org.junit ...

  5. 菜鸟学习Ado&period;net笔记一:Ado&period;net学习之SqlHelper类

    using System; using System.Collections.Generic; using System.Text; using System.Data.SqlClient; usin ...

  6. tomcat中的URL参数为中文,servlet接收后显示乱码

    URL中参数的值为中文时,servlet接收后显示为乱码,如下图: 这时候需要修改tomcat的中的server.xml文件.该文件路径为 tomcat安装目录下的conf文件夹.   为修改前的se ...

  7. 浅谈Vue不同场景下组件间的数据交流

    浅谈Vue不同场景下组件间的数据“交流”   Vue的官方文档可以说是很详细了.在我看来,它和react等其他框架文档一样,讲述的方式的更多的是“方法论”,而不是“场景论”,这也就导致了:我们在阅读完 ...

  8. poj3249 拓扑排序&plus;DP

    题意:给出一个有向无环图,每个顶点都有一个权值.求一条从入度为0的顶点到出度为0的顶点的一条路径,路径上所有顶点权值和最大. 思路:因为是无环图,则对于每个点经过的路径求其最大权值有,dp[i]=ma ...

  9. Java使用Jedis操作Redis大全

    Java操作Redis需要导入两个jar: commons-pool2-2.4.2.jar jedis-2.1.0.jar package com.chinasofti.test; import ja ...

  10. 转git的使用

    git的使用(包括创建远程仓库到上传代码到git的详细步骤以及git的一些常用命令) A创建远程仓库到上传代码到git 1)登陆或这注册git账号 https://github.com 2)创建远程仓 ...