【和春笋一起学C++】(五十三)使用指向对象的指针

目录

[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类为例:

  1. 使用常规表示法来声明指向对象的指针:
cpp 复制代码
MyClassString *st;
  1. 将指针初始化为指向已有的对象。示例程序中方法一使用的方式:
cpp 复制代码
MyClassString *st = &ar_st[0];
  1. 使用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");
  1. 指向对象的指针可以通过->操作符访问类方法,如示例代码中的st访问类的length方法:
cpp 复制代码
cout << "max length string: " << *st << ",  length is " << st->length() << endl;
  1. 可以对指向对象的指针应用解除引用操作符来获得对象,如示例代码中对st应用解除引用操作符:
cpp 复制代码
cout << "max length string: " << *st << ",  length is " << st->length() << endl;
相关推荐
云小逸13 小时前
【Nmap 设备类型识别技术】整体概况
服务器·c语言·网络·c++·nmap
liu****14 小时前
29.路径类dp
c++·算法·acm
阿猿收手吧!14 小时前
【C++】C++模板特化:精准定制泛型逻辑
开发语言·c++·算法
C语言小火车14 小时前
Qt样式实现方式详解:六大方法全面解析
c语言·c++·qt·学习
weixin_4521595514 小时前
C++与Java性能对比
开发语言·c++·算法
会叫的恐龙14 小时前
C++ 核心知识点汇总(第一日)(输入输出与变量、类型转换)
开发语言·c++
2301_7657031415 小时前
C++中的工厂模式实战
开发语言·c++·算法
退休钓鱼选手15 小时前
[CommonAPI + vsomeip]通信 客户端 5
c++·人工智能·自动驾驶
星火开发设计15 小时前
C++ 输入输出流:cin 与 cout 的基础用法
java·开发语言·c++·学习·算法·编程·知识
玖釉-15 小时前
探索连续细节层次(Continuous LOD):深入解析 NVIDIA 的 nv_cluster_lod_builder
c++·windows·图形渲染