【C++ Primer每日一刷之八】之九 创建动态数组

4.3.1. 创建动态数组

数组类型的变量有三个重要的限制:【数组长度固定不变】,【在编译时必须知道其长度】,【数组只在定义它的块语句内存在】。

实际的程序往往不能忍受这样的限制——它们需要在运行时动态地分配数组。

虽然数组长度是固定的,但动态分配的数组不必在编译时知道其长度,可以(通常也是)在运行时才确定数组长度。与数组变量不同,动态分配的数组将一直存在,直到程序显式释放它为止。每一个程序在执行时都占用一块可用的内存空间,用于存放动态分配的对象,此内存空间称为程序的自由存储区或堆。

C 语言程序使用一对标准库函数malloc 和 free 在自由存储区中分配存储空间,而 C++ 语言则使用 new 和delete表达式实现相同的功能。

动态数组的定义

【数组变量通过指定类型、数组名和维数来定义。而动态分配数组时,只需指定类型和数组长度,不必为数组对象命名,new 表达式返回指向新分配数组的第一个元素的指针】

int *pia = new int[10]; // array of 10uninitialized ints

此 new 表达式分配了一个含有 10个 int 型元素的数组,并返回指向该数组第一个元素的指针,此返回值初始化了指针 pia。new 表达式需要指定指针类型以及在方括号中给出的数组维数,该维数可以是任意的复杂表达式。【创建数组后,new 将返回指向数组第一个元素的指针在自由存储区中创建的数组对象是没有名字的,程序员只能通过其地址间接地访问堆中的对象。】

初始化动态分配的数组

动态分配数组时,如果数组元素具有类类型,将使用该类的默认构造函数(第2.3.4 节)实现初始化;如果数组元素是内置类型,则无初始化:

string *psa = new string[10]; // array of10 empty strings

int *pia = new int[10]; // array of 10uninitialized ints

这两个 new 表达式都分配了含有10 个对象的数组。其中第一个数组是 string类型,分配了保存对象的内存空间后,将调用 string 类型的默认构造函数依次初始化数组中的每个元素。第二个数组则具有内置类型的元素,分配了存储 10个 int 对象的内存空间,但这些元素没有初始化。也可使用跟在数组长度后面的一对空圆括号,对数组元素做值初始化(第

3.3.1 节):

int *pia2 = new int[10] ( ); // array of 10uninitialized ints

圆括号要求编译器对数组做值初始化,在本例中即把数组元素都设置为0。对于动态分配的数组,其元素只能初始化为元素类型的默认值,而不能像数组变量一样,用初始化列表为数组元素提供各不相同的初值。

const 对象的动态数组

如果我们在自由存储区中创建的数组存储了内置类型的 const 对象,则必须为这个数组提供初始化:因为数组元素都是 const 对象,无法赋值。实现这个要求的唯一方法是对数组做值初始化:

// error: uninitialized const array

const int *pci_bad = new const int[100];

// ok: value-initialized const array

const int *pci_ok = new const int[100]();

C++ 允许定义类类型的 const 数组,但该类类型必须提供默认构造函数:

// ok: array of 100 empty strings

const string *pcs = new const string[100];

在这里,将使用 string 类的默认构造函数初始化数组元素。当然,已创建的常量元素不允许修改——因此这样的数组实际上用处不大。

允许动态分配空数组

之所以要动态分配数组,往往是由于编译时并不知道数组的长度。我们可以编写如下代码:

size_t n = get_size( ); // get_size returnsnumber of elements needed

int* p = new int[n];

for (int* q = p; q != p + n; ++q)

/* process the array */ ;

计算数组长度,然后创建和处理该数组。有趣的是,如果 get_size 返回 0 则会怎么样?答案是:代码仍然正确执行。C++ 虽然不允许定义长度为 0 的数组变量,但明确指出,调用 new 动态创建长度为 0 的数组是合法的:

char arr[0]; // error: cannot definezero-length array

char *cp = new char[0]; // ok: but cp can‘tbe dereferenced

用 new 动态创建长度为 0 的数组时,new 返回有效的非零指针。该指针与new 返回的其他指针不同,不能进行解引用操作,因为它毕竟没有指向任何元素。而允许的操作包括:比较运算,因此该指针能在循环中使用;在该指针上加(减)0;或者减去本身,得 0 值。

在上述例题中,如果 get_size 返回 0,则仍然可以成功调用 new,但是p并没有指向任何对象,数组是空的。因为 n 为 0,所以 for 循环实际比较的是p 和 q,而 q 是用 p 初始化的,两者具有相等的值,因此 for 循环条件不成立,循环体一次都没有执行。

动态空间的释放

动态分配的内存最后必须进行释放,否则,内存最终将会逐渐耗尽。如果不再需要使用动态创建的数组,程序员必须显式地将其占用的存储空间返还给程序的自由存储区。C++ 语言为指针提供 delete [] 表达式释放指针所指向的数组空间:

delete [] pia;

该语句回收了 pia 所指向的数组,把相应的内存返还给自由存储区。在关键字 delete 和指针之间的空方括号对是必不可少的:它告诉编译器该指针指向的是自由存储区中的数组,而并非单个对象。

如果遗漏了空方括号对,这是一个编译器无法发现的错误,将导致程序在运行时出错。理论上,回收数组时缺少空方括号对,至少会导致运行时少释放了内存空间,从而产生内存泄漏(memory leak)。对于某些系统和/或元素类型,有可能会带来更严重的运行时错误。因此,在释放动态数组时千万别忘了方括号对。

C 风格字符串与 C++ 的标准库类型string 的比较

以下两段程序反映了使用 C 风格字符串与 C++ 的标准库类型 string的不同之处。使用 string 类型的版本更短、更容易理解,而且出错的可能性更小:

// C-style character string implementation

const char *pc = "a very long literalstring";

const size_t len = strlen(pc +1); // spaceto allocate

// performance test on string allocationand copy

for (size_t ix = 0; ix != 1000000; ++ix) {

char *pc2 = new char[len + 1]; // allocatethe space

strcpy(pc2, pc); // do the copy

if (strcmp(pc2, pc)) // use the new string

; // do nothing

delete [] pc2; // free the memory

}

// string implementation

string str("a very long literalstring");

// performance test on string allocationand copy

for (int ix = 0; ix != 1000000; ++ix) {

string str2 = str; // do the copy,automatically

allocated

if (str != str2) // use the new string

; // do nothing

}

// str2 is

automatically freed

这些程序将在4.3.1 节的习题中做进一步探讨。

动态数组的使用

通常是因为在编译时无法知道数组的维数,所以才需要动态创建该数组。例如,在程序执行过程中,常常使用char*指针指向多个C 风格字符串,于是必须根据每个字符串的长度实时地动态分配存储空间。采用这种技术要比建立固定大小的数组安全。如果程序员能够准确计算出运行时需要的数组长度,就不必再担心因数组变量具有固定的长度而造成的溢出问题。

假设有以下C 风格字符串:

const char *noerr = "success";

// ...

const char *err189 = "Error: afunction declaration must "

"specify a function returntype!";

我们想在运行时把这两个字符串中的一个复制给新的字符数组,于是可以用以下程序在运行时计算维数:

const char *errorTxt;

if (errorFound)

errorTxt = err189;

else

errorTxt = noerr;

// remember the 1 for the terminating null

int dimension = strlen(errorTxt) + 1;

char *errMsg = new char[dimension];

// copy the text for the error into errMsg

strncpy (errMsg, errorTxt, dimension);

别忘记标准库函数 strlen 返回的是字符串的长度,并不包括字符串结束符,在获得的字符串长度上必须加 1 以便在动态分配时预留结束符的存储空间。

时间: 2024-10-08 10:17:17

【C++ Primer每日一刷之八】之九 创建动态数组的相关文章

【C++ Primer每日一刷之八】之八 C 风格字符串

4.3 C 风格字符串 尽管 C++ 支持 C 风格字符串,但不应该在 C++ 程序中使用这个类型.C 风格字符串常常带来许多错误,是导致大量安全问题的根源. 在前面我们第一次使用了字符串字面值,并了解字符串字面值的类型是字符常量的数组,现在可以更明确地认识到:字符串字面值的类型就是const char 类型的数组.C++ 从 C 语言继承下来的一种通用结构是C 风格字符串,而字符串字面值就是该类型的实例.实际上,C 风格字符串既不能确切地归结为 C 语言的类型,也不能归结为 C++ 语言的类型

【C++ Primer每日一刷之九】创建动态数组

表达式 C++ 提供了丰富的操作符,并定义操作数为内置类型时,这些操作符的含义.除此之外,C++ 还支持操作符重载,允许程序员自定义用于类类型时操作符的含义.标准库正是使用这种功能定义用于库类型的操作符. 本章重点介绍 C++ 语言定义的操作符,它们使用内置类型的操作数:本章还会介绍一些标准库定义的操作符.第十四章将学习如何定义自己的重载操作符. 表达式由一个或多个操作数通过操作符组合而成.最简单的表达式仅包含一个字面值常量或变量.较复杂的表达式则由操作符以及一个或多个操作数构成. 每个表达式都

【C++ Primer每日一刷之七】指针操作

4.2.3 指针操作 指针提供间接操纵其所指对象的功能.与对迭代器进行解引用操作一样,对指针进行解引用可访问它所指的对象,* 操作符(解引用操作符)将获取指针所指的对象: string s("hello world"); string *sp = &s; // sp holds theaddress of s cout <<*sp; // prints hello world 对 sp 进行解引用将获得 s 的值,然后用输出操作符输出该值,于是最后一条语句输出了 s

【C++ Primer每日一刷之十二】 箭头操作符,条件操作符,sizeof操作符,逗号,优先级

5.6. 箭头操作符 C++ 语言为包含点操作符和解引用操作符的表达式提供了一个同义词:箭头操作符(->).点操作符(第 1.5.2 节)用于获取类类型对象的成员: item1.same_isbn(item2); // run thesame_isbn member of item1 如果有一个指向 Sales_item 对象的指针(或迭代器),则在使用点操作符 前,需对该指针(或迭代器)进行解引用: Sales_item *sp = &item1; (*sp).same_isbn(item

【C++ Primer每日一刷之五】标准库类型小结

标准库类型小结 C++ 标准库定义了几种更高级的抽象数据类型,包括 string 和 vector 类型.string 类型提供了变长的字符串,而 vector 类型则可用于管理同一类型 的对象集合.迭代器实现了对存储于容器中对象的间接访问.迭代器可以用于访问和遍历 string 类型和vectors 类型的元素.下一节将介绍 C++ 的内置数据类型:数组和指针.这两种类型提供了类似于 vector 和 string 标准库类型的低级抽象类型.总的来说,相对于C++ 内置数据类型的数组和指针而言

【C++ Primer每日一刷之十】 操作符(一)

表达式 C++ 提供了丰富的操作符,并定义操作数为内置类型时,这些操作符的含义.除此之外,C++ 还支持操作符重载,允许程序员自定义用于类类型时操作符的含义.标准库正是使用这种功能定义用于库类型的操作符. 本章重点介绍 C++ 语言定义的操作符,它们使用内置类型的操作数:本章还会介绍一些标准库定义的操作符.第十四章将学习如何定义自己的重载操作符. 表达式由一个或多个操作数通过操作符组合而成.最简单的表达式仅包含一个字面值常量或变量.较复杂的表达式则由操作符以及一个或多个操作数构成. 每个表达式都

【C++ Primer每日一刷之六】数组

引言 C++ 语言提供了两种类似于vector 和迭代器类型的低级复合类型--数组和指针.与vector 类型相似,数组也可以保存某种类型的一组对象:而它们的区别在于,数组的长度是固定的.数组一经创建,就不允许添加新的元素.指针则可以像迭代器一样用于遍历和检查数组中的元素. 现代 C++ 程序应尽量使用vector 和迭代器类型,而避免使用低级的数组和指针.设计良好的程序只有在强调速度时才在类实现的内部使用数组和指针. 数组是 C++ 语言中类似于标准库vector 类型的内置数据结构.与 ve

【C++ Primer每日刷】之一 迭代器

迭代器的介绍 概述 迭代器是一种检查容器内元素并遍历元素的数据类型. 迭代器(iterator)是一种对象,它能够用来遍历标准模板库容器中的部分或全部元素,每个迭代器对象代表容器中的确定的地址.迭代器修改了常规指针的接口,所谓迭代器是一种概念上的抽象:那些行为上像迭代器的东西都可以叫做迭代器.然而迭代器有很多不同的能力,它可以把抽象容器和通用算法有机的统一起来. 标准库为每一种标准容器(包括 vector)定义了一种迭代器类型.迭代器类型提供了比下标操作更通用化的方法:所有的标准库容器都定义了相

每日算法之二十九:Search in Rotated Sorted Array

在一个经过旋转后的有序数组中查找一个目标元素. Suppose a sorted array is rotated at some pivot unknown to you beforehand. (i.e., 0 1 2 4 5 6 7 might become 4 5 6 7 0 1 2). You are given a target value to search. If found in the array return its index, otherwise return -1.