C++指针引用详解

引言

实际开发中遇到的问题,将问题简化如下:

  1. 在外部定义一个空指针
  2. 将该空指针值传递进A函数中,在A函数中赋值
  3. 赋值完以后,需要在B函数中使用该指针,但该指针却是空的

上面的说法可能不够准确,直接看实际的例子吧

#include <iostream>
#include <string>
#include <list>
#include <vector>
using namespace std;

class Person
{
public:
	Person(const string name, const int age, const int gender) 
		:m_name(name), m_age(age), m_gender(gender) 
	{
		cout << "call constructor." << endl;
	};

	~Person() { cout << "call deconstructor." << endl; };

	void printPersonInfo();

private:
	string m_name;
	int m_age;
	int m_gender;
};

void Person::printPersonInfo()
{
	cout << "name:" << m_name << ", age:" << m_age << ", gender:" << (m_gender == 0 ? "female" : "male") << endl;
}

void getPerson(const vector<Person*>& persons, const int index, Person* searchPerson)
{
	if (index < 0 || index >= persons.size()) {
		cout << "index is invalid." << endl;
		return;
	}

	searchPerson = persons[index];
	searchPerson->printPersonInfo();

	cout << searchPerson << endl;
	cout << persons[index] << endl;
}

void doSomething(Person* person)
{
	if (person == nullptr) {
		cout << "Person not exist." << endl;
		return;
	}

	person->printPersonInfo();
	cout << person << endl;
}

int main()
{
	vector<Person*> persons;
	Person* person0 = new Person("Wang", 18, 0);
	persons.emplace_back(person0);
	Person* person1 = new Person("Zhao", 21, 1);
	persons.emplace_back(person1);
	Person* person2 = new Person("Qian", 24, 0);
	persons.emplace_back(person2);
	cout << persons.size() << endl;

	Person* searchPerson = nullptr;
	getPerson(persons, 0, searchPerson);
	doSomething(searchPerson);
	cout << searchPerson << endl;

	for (int i = 0; i < persons.size(); ++i)
	{
		delete persons[i];
		persons[i] = nullptr;
	}

	return 0;
}

在上面的例子中,新建了Person类,流程如下:

  1. 构造一个persons指针数组
  2. 构造一个空的Person指针searchPerson
  3. getPerson函数中查找匹配的指针并赋值给searchPerson
  4. doSomething中操作该指针,这里只是简单的个人信息打印printPersonInfo

上面的代码运行结果如下

call constructor.
call constructor.
call constructor.
3
name:Wang, age:18, gender:female
00000281A88A4FC0
00000281A88A4FC0
Person not exist.
0000000000000000
call deconstructor.
call deconstructor.
call deconstructor.

可以看到getPerson函数中searchPerson的地址是00000281A88A4FC0

但是在doSomething函数中操作时却是空的,这实在很郁闷

问题分析

问题的根本还是对指针和函数的理解不够清晰

  • 指针本身也就是4个字节的变量,只不过这个变量内部存储的是地址而已
  • 函数按值传递,内部形参的改变不会影响外部的实参

弄清楚上面的两点,问题就比较明显了
getPerson 函数中,searchPerson 是按值传递的(指针按值传递),对它的修改不会影响调用者的值

void getPerson(const vector<Person*>& persons, const int index, Person* searchPerson)

searchPerson = persons[index]; 的修改仅限于 getPerson 函数内部,调用者中的 searchPerson 仍然是 nullptr
searchPerson 改为按引用传递:

void getPerson(const vector<Person*>& persons, const int index, Person*& searchPerson)

这样 searchPerson 的值在函数内部修改后,会正确反映到调用者的上下文中

调整后的打印输出如下

call constructor.
call constructor.
call constructor.
3
name:Wang, age:18, gender:female
0000015F986F4FC0
0000015F986F4FC0
name:Wang, age:18, gender:female
0000015F986F4FC0
0000015F986F4FC0
call deconstructor.
call deconstructor.
call deconstructor.

可以看到searchPerson已经非空,可以在doSomething中正常操作

总结

指针引用是指引用一个指针,即一个指针变量本身作为引用的目标。这种技术允许修改指针本身(即让它指向其他地址)而不仅仅是修改它指向的内容。通过指针引用,可以在函数中直接改变调用者传入的指针的值
指针引用的主要用途包括:

  1. 修改指针的值(指向的地址)
    在函数内部修改指针,使其指向不同的内存地址,并让这种修改在调用者中生效
  2. 避免多级指针的复杂性
    如果没有指针引用,修改指针值可能需要使用二级指针;使用指针引用可以简化代码逻辑
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Prejudices

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值