目录
[1 指向对象的指针使用示例](#1 指向对象的指针使用示例)
[2 小结](#2 小结)
1 指向对象的指针使用示例
使用《C++中string类的简介》文章中声明的MyClassString类,定义一个MyClassString类对象的数组,找出数组中字符串长度最大的MyClassString对象,这里使用一个指向MyClassString类对象的指针来跟踪数组中长度最长的字符串,代码如下:
cpp
////方法一
int main()
{
using namespace std;
MyClassString ar_st[] = { "rice","food","cat","home","school","cake","dog" };
size_t length = sizeof(ar_st) / sizeof(ar_st[0]);
MyClassString *st = &ar_st[0];
for (int i = 0; i < length; i++)
{
if (ar_st[i].length() > st->length())
{
st = &ar_st[i];
}
}
cout << "max length string: " << *st << ", length is " << st->length() << endl;
return 0;
}
程序说明:程序定义一个指向MyClassString类对象的指针,并将它指向数组中第一个元素的地址,通过for循环逐个比较数组中字符串长度,找出长度最大的字符串,因st为指向MyClassString对象的指针,因此可以使用->操作符访问类方法。
方法二:还可以使用new操作符来定义一个指向MyClassString对象的指针。
cpp
///方法二
int main()
{
using namespace std;
MyClassString ar_st[] = { "rice","food","cat","home","school","cake","dog" };
size_t length = sizeof(ar_st) / sizeof(ar_st[0]);
MyClassString *st = new MyClassString;
for (int i = 0; i < length; i++)
{
if (ar_st[i].length() > st->length())
{
st = &ar_st[i];
}
}
cout << "max length string: " << *st << ", length is " << st->length() << endl;
return 0;
}
程序说明:上面代码中虽然使用了new操作符,但不能使用delete来释放st所指向的内存空间,原因如下:
cpp
MyClassString *st = new MyClassString;
该语句的含义是使用默认构造函数创建一个未被命名的对象,并使st指向该对象。
在for循环语句中st由指向未被命名的对象变成指向ar_st数组中的某一个元素,如果再使用delete释放st所指向的内存空间,将引发错误,因为此时st指向了ar_st数组中的某一个元素,**而ar_st数组中的元素存放在栈上,使用delete释放栈上的内存空间则必然导致程序运行崩溃。**所以在该程序中的for循环之后不能使用delete释放st所指向的内存空间。
在该程序中还可以将语句MyClassString *st = new MyClassString;换成如下语句:
cpp
MyClassString *st = new MyClassString(ar_st[0]);
上面语句表示使用构造函数MyClassString(const MyClassString& st)来创建一个未被命名的对象,并使st指向该对象。此处的new分配的内存空间是用来存储对象的,用来保存str指针的地址以及len成员,并不会保存str所指向的字符串。而创建一个未被命名的对象时,将调用构造函数,构造函数将使用new分配用于存储字符串的内存空间,并将字符串的地址赋给str。同样,当使用delete释放st时,释放的是存储str指针地址和len成员的空间,并不释放str指向的内存空间,而该任务将由对象的析构函数来完成。
析构函数在下面几种情况下都会被调用:
- 如果对象是动态变量,则当执行完定义该对象的程序块时,将调用该对象的析构函数;
- 如果对象是静态变量,则在程序结束时将调用对象的析构函数。
- 如果对象是用new创建的,则仅当显式使用delete删除对象时,其析构函数才会被调用;
所以方法二会造成内存泄露,未被命名的对象所占据的内存空间没有被释放,同样也没有调用该对象的析构函数。
2 小结
以《C++中string类的简介》文章中声明的MyClassString类为例:
- 使用常规表示法来声明指向对象的指针:
cpp
MyClassString *st;
- 将指针初始化为指向已有的对象。示例程序中方法一使用的方式:
cpp
MyClassString *st = &ar_st[0];
- 使用new来初始化指针,这将创建一个新的对象,示例程序中方法二使用的方式:
cpp
MyClassString *st = new MyClassString(ar_st[0]);
注:对类使用new将调用相应的类构造函数来初始化新创建的对象。
cpp
///调用默认构造函数创建新对象
MyClassString *st = new MyClassString;
////调用复制构造函数创建新对象:MyClassString(const MyClassString& st);
MyClassString *st = new MyClassString(ar_st[0]);
////调用该构造函数创建新对象:MyClassString(const char* s);
MyClassString *st = new MyClassString("school");
- 指向对象的指针可以通过->操作符访问类方法,如示例代码中的st访问类的length方法:
cpp
cout << "max length string: " << *st << ", length is " << st->length() << endl;
- 可以对指向对象的指针应用解除引用操作符来获得对象,如示例代码中对st应用解除引用操作符:
cpp
cout << "max length string: " << *st << ", length is " << st->length() << endl;