【和春笋一起学C++】(五十二)关于函数返回对象时的注意事项

目录

[1. 返回对象](#1. 返回对象)

[2. 返回指向对象的引用](#2. 返回指向对象的引用)

[3. 返回const对象](#3. 返回const对象)

[4. 返回指向对象的const引用](#4. 返回指向对象的const引用)

[5. 总结](#5. 总结)


函数返回对象有以下几种方式:

  1. 返回对象;
  2. 返回指向对象的引用;
  3. 返回const对象;
  4. 返回指向对象的const引用;

1. 返回对象

函数返回对象需要调用复制构造函数,而返回对象的引用则不需要,显然,返回对象的引用执行效率更高。但如果被返回的对象是调用函数中的局部变量,则不能按引用方式返回它,因为调用函数执行完毕时,局部对象将调用其析构函数。因此,当控制权回到调用函数时,引用指向的对象将不再存在。在这种情况下,应返回对象,而不是引用。如下面的strcat函数,只能返回string对象,不能返回string对象引用。

cpp 复制代码
std::string strcat(std::string& s1, std::string& s2)
{
	std::string s3 = s1 + s2;
	return s3;
}

int main()
{
	string s1 = "abc";
	string s2 = "efg";
	string s3 = strcat(s1, s2);
	cout << "s3=" << s3 << endl;
    return 0;
}

如果把strcat函数的返回改成string对象的引用,则运行将报错,因为strcat函数中的s3是局部变量,在strcat函数运行结束后s3对象将不再存在。但如果返回的是string对象,就不会出现这种问题,s3会通过复制构造函数创建一个临时对象,并将s3的内容拷贝至该临时对象,调用函数通过访问临时对象获取具体的数据内容。

2. 返回指向对象的引用

如《C++中string类简介》文章中的重载赋值操作符函数,定义如下:

cpp 复制代码
MyClassString& MyClassString::operator=(const MyClassString& s)
{
	if (this == &s)
	{
		return *this;
	}
	delete[]str;
	len = s.len;
	str = new char[len + 1];
	strcpy(str,s.str);
	return *this;
}

函数参数为指向MyClassString对象的const引用,返回指向MyClassString对象的引用。函数返回传递给它的对象的引用,这种方式在操作符重载函数中很常见。MyClassString类的输出操作符重载函数也返回了传递给它的对象的引用,如下:

cpp 复制代码
friend std::ostream& operator<< (std::ostream& os, const MyClassString& st)
{
	os << st.str;
	return os;
}

函数返回对象的引用可以提高函数的运行效率,但使用对象的引用作为函数返回时有一些限制条件,即返回不能是局部对象。

3. 返回const对象

返回const对象可以防止返回值被随意篡改,如上面的strcat函数,将返回值改成const,如下:

cpp 复制代码
const std::string strcat(std::string& s1, std::string& s2)
{
	std::string s3 = s1 + s2;
	return s3;
}

此时,下面的代码成立:

cpp 复制代码
int main()
{
	string s1 = "abc";
	string s2 = "efg";
	string s3 = strcat(s1, s2);
	cout << "s3=" << s3 << endl;
    return 0;
}

但如下的代码不成立,不能编译通过。

cpp 复制代码
int main()
{
	string s1 = "abc";
	string s2 = "efg";
	string s3 = "xyz";
	strcat(s1, s2) = s3;
    return 0;
}

因为const对象是不能被赋值的。

4. 返回指向对象的const引用

返回指向对象的const引用可以提高程序的执行效率,如将上述strcat函数改成指向string对象的const引用,如下:

cpp 复制代码
const std::string& strcat(std::string& s1, std::string& s2)
{
	s1 = s1 + s2;
	return s1;
}

5. 总结

  1. 如果方法或函数要返回局部对象,则应返回对象,而不是指向对象的引用。在这种情况下,将使用复制构造函数来生成返回的对象。
  2. 如果方法或函数要返回一个没有公有复制构造函数的类(如ostream类)的对象,则它必须返回一个指向该类对象的引用。
  3. 如果方法或函数可以返回对象,也可以返回指向对象的引用,这种情况下,应优先选择返回对象的引用,因为其执行效率更高。
相关推荐
2401_8920709813 小时前
【Linux C++ 日志系统实战】LogFile 日志文件管理核心:滚动策略、线程安全与方法全解析
linux·c++·日志系统·日志滚动
yuzhuanhei13 小时前
Visual Studio 配置C++opencv
c++·学习·visual studio
不爱吃炸鸡柳13 小时前
C++ STL list 超详细解析:从接口使用到模拟实现
开发语言·c++·list
十五年专注C++开发14 小时前
RTTR: 一款MIT 协议开源的 C++ 运行时反射库
开发语言·c++·反射
‎ദ്ദിᵔ.˛.ᵔ₎14 小时前
STL 栈 队列
开发语言·c++
2401_8920709814 小时前
【Linux C++ 日志系统实战】高性能文件写入 AppendFile 核心方法解析
linux·c++·日志系统·文件写对象
郭涤生14 小时前
STL vector 扩容机制与自定义内存分配器设计分析
c++·算法
༾冬瓜大侠༿14 小时前
vector
c语言·开发语言·数据结构·c++·算法
cccyi714 小时前
【C++ 脚手架】etcd 的介绍与使用
c++·服务发现·etcd·服务注册
liu****14 小时前
第16届省赛蓝桥杯大赛C/C++大学B组(京津冀)
开发语言·数据结构·c++·算法·蓝桥杯