《C++编程风格(修订版)》——2.5 动态内存的一致性

简介:

本节书摘来自异步社区出版社《C++编程风格(修订版)》一书中的第2章,第2.5节,作者:【美】Tom Cargill,更多章节内容可以访问云栖社区“异步社区”公众号查看。

2.5 动态内存的一致性

C++编程风格(修订版)
在程序清单 2.2 的 string 类中仍然存在着一些问题和不一致的地方。其中,在动态内存管理 上的不一致性与我们在前面所看到的不一致性是一样的,都是严重的问题。对于所有动态分配的 内存,我们都需要回答两个问题:首先,动态内存是不是足够大以容纳将要存储的信息?其次, 是不是所有的动态内存都是可回收的?

在默认构造函数中分配的字符数组肯定可以容纳空字符串:
image

这个构造函数所基于的假设是:在创建对象时将会为字符串分配内存,并且这个内存足以 容纳在对象生存期内需要保存的任意字符串。成员函数 assign() 与这个假设也是一致的:
image

在 assign() 中调用 strcpy() 对参数字符串进行拷贝时,并没有考虑到目标字符数组的长度或 者大小。编写客户代码的程序员必须保证——在创建对象时,无论调用的是哪个构造函数——在 构造函数中所创建的数组必须能够容纳在 assign() 中复制的任意字符串。

然而,在成员函数 concat() 中采用了一种不同的方法:在创建每个字符串时,总是动态地决 定所需数组的精确大小。函数 concat() 忽略了在创建 string 对象时已经分配好的字符数组,即使 这个已分配的数组是足够大的:

image

在 assign() 和 concat() 这两个函数的表现行为上存在着不一致性。它们的区别在于,在为 string

对象设置新值时,是否会动态分配字符数组:assing() 永远不会分配,而 concat() 则总是会分配。

接口一致性
上面哪种控制数组大小的方法是更好的?和许多软件决策一样,没有哪种方法是绝对的“正 确”或者绝对的“错误”。这两种方法都有各自的优点。保持在构造函数中分配的数组不变(assign() 中的做法)是一种高效的方法,因为在后续的操作中就无需再调用内存分配函数。对每个字符串 值都动态地决定数组的大小(concat() 中的做法)则是一种更安全的方法,因为这种方法杜绝了 数组的“越界”行为。

这两种方法都可以用在类中,但我们只能使用其中的一种,以保持类一致性,而不应该将 这两种方法混合使用。否则,在使用这个类时,程序员将不得不去了解在接口中不同操作之间的 不同约定。如果一个程序员只使用过 concat(),并且知道了数组的大小是动态增长的,那么他就 会假定 assign() 也是同样的行为,因此,当在 assign() 中发生数组内存的越界问题时,他所感到 的沮丧应该是可以预见的。

类的接口定义应该是一致的——避免产生困惑。

相关文章
|
存储 程序员 编译器
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
|
11月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
384 26
|
编译器 C++ 开发者
【C++】深入解析C/C++内存管理:new与delete的使用及原理(三)
【C++】深入解析C/C++内存管理:new与delete的使用及原理
380 3
|
C语言 C++
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
216 1
|
存储 缓存 编译器
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
796 68
|
存储 Linux C语言
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
|
安全 C语言 C++
彻底摘明白 C++ 的动态内存分配原理
大家好,我是V哥。C++的动态内存分配允许程序在运行时请求和释放内存,主要通过`new`/`delete`(用于对象)及`malloc`/`calloc`/`realloc`/`free`(继承自C语言)实现。`new`分配并初始化对象内存,`delete`释放并调用析构函数;而`malloc`等函数仅处理裸内存,不涉及构造与析构。掌握这些可有效管理内存,避免泄漏和悬空指针问题。智能指针如`std::unique_ptr`和`std::shared_ptr`能自动管理内存,确保异常安全。关注威哥爱编程,了解更多全栈开发技巧。 先赞再看后评论,腰缠万贯财进门。
679 0
|
存储 程序员 编译器
简述 C、C++程序编译的内存分配情况
在C和C++程序编译过程中,内存被划分为几个区域进行分配:代码区存储常量和执行指令;全局/静态变量区存放全局变量及静态变量;栈区管理函数参数、局部变量等;堆区则用于动态分配内存,由程序员控制释放,共同支撑着程序运行时的数据存储与处理需求。
811 22
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
365 3
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
833 4