引用与指针经常混淆,总结一下
文章目录
-
- [1. 引用与指针的区别](#1. 引用与指针的区别)
- [2. 引用传递数组](#2. 引用传递数组)
- [3. 通过引用传递容器和类](#3. 通过引用传递容器和类)
- [4. 多线程传递容器时用 std:: ref 替代引用传递](#4. 多线程传递容器时用 std:: ref 替代引用传递)
1. 引用与指针的区别
- 引用(Reference):引用是变量的别名,本质上不是一个变量,而是给一个已经存在的变量起的一个别名。
- 本质:引用就是变量的别名,不占用新的内存空间。
- 指针(Pointer):指针是一个变量,存放的是另一个变量的内存地址,需要通过解引用 * 操作符访问该地址的内容。
- 本质:指针就是存放地址的变量。
cpp
int a = 10;
int* p = &a; // p存储的是a的地址
cout << *p; // 解引用,输出a的值
cpp
int a = 10;
int& b = a; // b是a的别名
b = 20;
cout << a; // 输出20
内存结构:
变量 | 地址 | 内容 |
---|---|---|
a | 0x1000 | 20 |
b | 0x1000 | 20 |
-
b 只是 a 的别名,它们共享同一个内存空间,所以修改 b 就是修改 a。
-
引用一旦绑定,就不能再改,而指针则不是
cpp
int a = 10;
int b = 20;
int& r = a; // r是a的别名
r = b; // ❌ 不是修改引用,而是给a赋值
- 引用不占用额外内存,它就是变量的别名。
2. 引用传递数组
- 通过引用可以传递整个数组,避免数组退化成指针。
数组的数组名(array name)本质上是数组首元素的地址,所以当数组传参时,传递的实际上是数组首元素的地址,而不是整个数组。
cpp
void func(int* arr);
void func(int arr[]); // 与前面等价
二者等价,因为数组会自动退化为指针。
例子:通过指针传递一维数组
cpp
#include <iostream>
using namespace std;
void modifyArray(int* arr, int size) {
for(int i = 0; i < size; i++) {
arr[i] += 10;
}
}
int main() {
int numbers[] = {1, 2, 3, 4, 5};
int size = sizeof(numbers) / sizeof(numbers[0]);
modifyArray(numbers, size);
for(int i = 0; i < size; i++) {
cout << numbers[i] << " ";
}
return 0;
}
- 传入的是数组首元素的地址,即 int* arr 接收的是 &numbers[0] 的地址
通过引用传递一维数组
cpp
#include <iostream>
using namespace std;
void modifyArray(int (&arr)[5]) {
for(int i = 0; i < 5; i++) {
arr[i] += 10;
}
}
int main() {
int numbers[] = {1, 2, 3, 4, 5};
modifyArray(numbers);
for(int i = 0; i < 5; i++) {
cout << numbers[i] << " ";
}
return 0;
}
- int (&arr)[5] 表示引用一个长度为5的数组。
- 传入时,整个数组的内存地址传递过来,并且保留数组大小。
通过指针传递二维数组的例子:
cpp
#include <iostream>
using namespace std;
void printArray(int (*arr)[4], int rows) {
for(int i = 0; i < rows; i++) {
for(int j = 0; j < 4; j++) {
cout << arr[i][j] << " ";
}
cout << endl;
}
}
int main() {
int numbers[3][4] = {
{1, 2, 3, 4},
{5, 6, 7, 8},
{9, 10, 11, 12}
};
printArray(numbers, 3);
return 0;
}
通过引用传递二维数组的例子:
cpp
#include <iostream>
using namespace std;
void printArray(int (&arr)[3][4]) {
for(int i = 0; i < 3; i++) {
for(int j = 0; j < 4; j++) {
cout << arr[i][j] << " ";
}
cout << endl;
}
}
int main() {
int numbers[3][4] = {
{1, 2, 3, 4},
{5, 6, 7, 8},
{9, 10, 11, 12}
};
printArray(numbers);
return 0;
}
3. 通过引用传递容器和类
如果传递类对象(class),也应该使用引用传递,否则会拷贝整个对象,非常消耗资源。
cpp
#include <iostream>
using namespace std;
// ✅ 定义一个 Person 类
class Person {
public:
string name;
int age;
Person(string name, int age) {
this->name = name;
this->age = age;
}
void display() {
cout << "Name: " << name << ", Age: " << age << endl;
}
};
// ✅ 通过引用传递类对象
void modifyPerson(Person& p) {
p.age += 10;
p.name = "Mr. " + p.name;
}
int main() {
Person person("John", 25);
// 传递引用,避免拷贝对象
modifyPerson(person);
// 输出修改后的信息
person.display();
return 0;
}
对于一些容器,可以通过引用传递,避免拷贝且可以修改容器中内容。
cpp
#include <iostream>
#include <vector>
using namespace std;
// ✅ 通过引用传递 vector
void modifyVector(vector<int>& vec) {
for(int& val : vec) {
val *= 2;
}
}
int main() {
vector<int> numbers = {1, 2, 3, 4, 5};
// 传递 vector 的引用
modifyVector(numbers);
// 打印修改后的 vector
for(int val : numbers) {
cout << val << " ";
}
return 0;
}
4. 多线程传递容器时用 std:: ref 替代引用传递
C++ STL 容器 (如 vector、list) 在多线程或函数包装中,默认是按值传递。
cpp
#include <iostream>
#include <vector>
#include <thread>
using namespace std;
void modifyVector(vector<int>& vec) {
for (auto& v : vec) {
v *= 2;
}
}
int main() {
vector<int> nums = {1, 2, 3, 4, 5};
// 将函数和容器传入线程
thread t(modifyVector, nums); // ❌ 出问题!
t.join();
// 输出结果
for (auto v : nums) {
cout << v << " ";
}
return 0;
}
- 虽然 modifyVector 的参数是 vector&,但是 std::thread 默认是按值传递!
- thread 会拷贝一份 nums,导致无法修改原容器。
C++ 提供了一个工具 std::ref,专门用于"强制引用传递",防止容器被拷贝。
cpp
#include <iostream>
#include <vector>
#include <thread>
#include <functional> // 包含 std::ref
using namespace std;
void modifyVector(vector<int>& vec) {
for (auto& v : vec) {
v *= 2;
}
}
int main() {
vector<int> nums = {1, 2, 3, 4, 5};
// ✅ 使用 std::ref,强制引用传递
thread t(modifyVector, std::ref(nums));
t.join();
// 输出结果
for (auto v : nums) {
cout << v << " ";
}
return 0;
}
在 C++ 中,以下场景:
- std::thread
- std::function
- std::bind
- std::async
- std::packaged_task
都默认按值传递参数。如果你传入容器、类、数组、函数等复杂对象,会直接拷贝副本,而不是传引用,需要用 std::ref