19、C++ Primer 4th 笔记,类管理指针成员

简介: 1、一般采取三种方法 1)指针成员采取常规指针型行为。这样的类具有指针的所有缺陷但无需特殊的复制控制。 2)类可以实现所谓的“智能指针”行为。指针所指向的对象是共享的,但类能够防止悬垂指针。 3)类采取值型行为。

1、一般采取三种方法

1)指针成员采取常规指针型行为。这样的类具有指针的所有缺陷但无需特殊的复制控制。

2)类可以实现所谓的“智能指针”行为。指针所指向的对象是共享的,但类能够防止悬垂指针。

3)类采取值型行为。指针所指向的对象是唯一的,由每个类对象独立管理。

2、类中定义指针时,需要考虑的有:类的复制,类的赋值,类的析构,共享对象,及悬垂指针。

3、定义智能指针

一个行为类似指针但也提供其他功能的类。智能指针的一个通用形式是接受指向动态分配对象的指针并负责删除该对象。用户分配对象,但由智能指针类删除它。智能指针类需要实现复制控制成员来管理指向共享对象的指针。引用计数是实现智能指针的常用方法。

注意:是由构造函数设置共享对象的状态并将使用计数置为1

1)引入使用计数

定义智能指针的通用技术是采用一个使用计数(use count,也称为引用计数,reference count)。智能指针类将一个计数器与类中成员指向的对象相关联。使用计数为0时,删除对象。

对一个对象进行赋值时,赋值操作符减少左操作数对象的使用计数的值(如果使用计数减至0,则删除对象),并增加右操作数对象的使用计数的值。

2)实现使用计数有两种策略。这里引入一种:定义一个单独的具体类用以封装使用计数和相关指针。

示例

class U_Ptr
{
	friend class HasPtr;
	int *ip;
	size_t use;
	U_Ptr(int *p):ip(p), use(1){}
	~U_Ptr() {delete ip;}
};

class HasPtr
{
public:
	HasPtr(int *p, int i):ptr(new U_Ptr(p)), val(i) {}
	HasPtr(const HasPtr& orig):ptr(orig.ptr), val(orig.val)
	{
		++ptr->use;
	}
	HasPtr& operator=(const HasPtr&);
	~HasPtr()
	{
		if (--ptr->use == 0)
			delete ptr;
	}
private:
	U_Ptr *ptr;
	int val;
};

HasPtr& HasPtr::operator=(const HasPtr& rhs)
{
	++rhs.ptr->use;
	if (--ptr->use == 0)
		delete ptr;
	ptr = rhs.ptr;
	val = rhs.val;
	return *this;
}

4、定义值型类

与处理智能指针不同的是,给指针成员提供值语义。具有值语义的类所定义的对象,其行为很像算术类型的对象:复制值型对象时,会得到一个不同的新副本。当发生复制时,改变的是指针所指向的值,而不是指针。

示例

class HasPtr
{
public:
	HasPtr(const int &p, int i):ptr(new int(p)), val(i){}
	HasPtr(const HasPtr &orig):ptr(new int(*orig.ptr)), val(orig.val){}
	HasPtr& operator=(const HasPtr&);
	~HasPtr() {delete ptr;}
private:
	int *ptr;
	int val;
};

HasPtr& HasPtr::operator=(const HasPtr& rhs)
{
	*ptr = *rhs.ptr; //是赋值
	val = rhs.val;
	return *this;
}

参考

[1] http://blog.163.com/zhoumhan_0351/blog/static/3995422720100250413207/

[2] http://blog.163.com/zhoumhan_0351/blog/static/39954227201032845132592/

[3] http://blog.163.com/zhoumhan_0351/blog/static/399542272010318112048522/

[4] http://blog.163.com/zhoumhan_0351/blog/static/39954227201032092854732/

[5] http://blog.163.com/zhoumhan_0351/blog/static/3995422720100284731826/

[6] http://blog.163.com/zhoumhan_0351/blog/static/39954227201012465955824/

目录
相关文章
|
3月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
83 0
|
3月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
164 0
|
5月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
161 12
|
6月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
指针进阶(C语言终)
指针进阶(C语言终)
|
10月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
831 13
|
存储 人工智能 C语言
C语言程序设计核心详解 第八章 指针超详细讲解_指针变量_二维数组指针_指向字符串指针
本文详细讲解了C语言中的指针,包括指针变量的定义与引用、指向数组及字符串的指针变量等。首先介绍了指针变量的基本概念和定义格式,随后通过多个示例展示了如何使用指针变量来操作普通变量、数组和字符串。文章还深入探讨了指向函数的指针变量以及指针数组的概念,并解释了空指针的意义和使用场景。通过丰富的代码示例和图形化展示,帮助读者更好地理解和掌握C语言中的指针知识。
428 4
|
11月前
|
C语言
无头链表二级指针方式实现(C语言描述)
本文介绍了如何在C语言中使用二级指针实现无头链表,并提供了创建节点、插入、删除、查找、销毁链表等操作的函数实现,以及一个示例程序来演示这些操作。
136 0
|
编译器 C语言
【C语言初阶】指针篇—下
【C语言初阶】指针篇—下
|
存储 C语言
【C语言初阶】指针篇—上
【C语言初阶】指针篇—上