引用变量是一个别名,也就是说,它是某个已存在变量的另一个名字。一旦把引用初始化为某个变量,就可以使用该引用名称或变量名称来指向变量

C++ 引用 vs 指针

引用很容易与指针混淆,它们之间有三个主要的不同:

  • 不存在空引用。引用必须连接到一块合法的内存。
  • 一旦引用被初始化为一个对象,就不能被指向到另一个对象。指针可以在任何时候指向到另一个对象。
  • 引用必须在创建时被初始化。指针可以在任何时间被初始化。

C++ 中创建引用

试想变量名称是变量附属在内存位置中的标签,您可以把引用当成是变量附属在内存位置中的第二个标签。因此,您可以通过原始变量名称或引用来访问变量的内容。例如:

  1. int i = 17;

我们可以为 i 声明引用变量,如下所示:

  1. int& r = i;
  2. double& s = d;

在这些声明中,& 读作引用。因此,第一个声明可以读作 “r 是一个初始化为 i 的整型引用”,第二个声明可以读作 “s 是一个初始化为 d 的 double 型引用”。下面的实例使用了 int 和 double 引用:

  1. #include <iostream>
  2. using namespace std;
  3. int main ()
  4. {
  5. // 声明简单的变量
  6. int i;
  7. double d;
  8. int i1;
  9. // 声明引用变量
  10. int& r = i;
  11. double& s = d;
  12. i = 5;
  13. cout << "Value of i : " << i << endl;
  14. cout << "Value of i reference : " << r << endl;
  15. d = 11.7;
  16. cout << "Value of d : " << d << endl;
  17. cout << "Value of d reference : " << s << endl;
  18. i = 66;
  19. d = 666.66;
  20. cout << "修改后的 i : " << i << endl;
  21. cout << "修改后的 d : " << d << endl;
  22. cout << "修改后的 i reference : " << r << endl;
  23. cout << "修改后的 d reference : : " << s << endl;
  24. r = 88;
  25. s = 888.888;
  26. cout << "修改后的 i : " << i << endl;
  27. cout << "修改后的 d : " << d << endl;
  28. cout << "修改后的 i reference : " << r << endl;
  29. cout << "修改后的 d reference : : " << s << endl;
  30. return 0;
  31. }
  32. /*
  33. Value of i : 5
  34. Value of i reference : 5
  35. Value of d : 11.7
  36. Value of d reference : 11.7
  37. 修改后的 i : 66
  38. 修改后的 d : 666.66
  39. 修改后的 i reference : 66
  40. 修改后的 d reference : : 666.66
  41. 修改后的 i reference : 88
  42. 修改后的 d reference : : 888.888
  43. */

引用通常用于函数参数列表和函数返回值。下面列出了 C++ 程序员必须清楚的两个与 C++ 引用相关的重要概念:

概念 描述
把引用作为参数 C++ 支持把引用作为参数传给函数,这比传一般的参数更安全
把引用作为返回值 可以从 C++ 函数中返回引用,就像返回其他数据类型一样

把引用作为参数

我们已经讨论了如何使用指针来实现引用调用函数。下面的实例使用了引用来实现引用调用函数

  1. #include <iostream>
  2. using namespace std;
  3. // 函数声明
  4. void swap(int& x, int& y);
  5. int main ()
  6. {
  7. // 局部变量声明
  8. int a = 100;
  9. int b = 200;
  10. cout << "交换前,a 的值:" << a << endl;
  11. cout << "交换前,b 的值:" << b << endl;
  12. swap(a, b); // 调用函数来交换值
  13. cout << "交换后,a 的值:" << a << endl;
  14. cout << "交换后,b 的值:" << b << endl;
  15. return 0;
  16. }
  17. // 函数定义
  18. void swap(int& x, int& y)
  19. {
  20. int temp;
  21. temp = x; // 保存地址 x 的值
  22. x = y; // 把 y 赋值给 x
  23. y = temp; // 把 x 赋值给 y
  24. return;
  25. }
  26. /*
  27. 交换前,a 的值:100
  28. 交换前,b 的值:200
  29. 交换后,a 的值:200
  30. 交换后,b 的值:100
  31. */

把引用作为返回值

通过使用引用来替代指针,会使 C++ 程序更容易阅读和维护。C++ 函数可以返回一个引用,方式与返回一个指针类似。
当函数返回一个引用时,则返回一个指向返回值的隐式指针。这样,函数就可以放在赋值语句的左边。例如,请看下面这个简单的程序:

  1. #include <iostream>
  2. using namespace std;
  3. double vals[] = {1.1, 2.2, 3.3};
  4. double& setValues(int i) {
  5. double& ref = vals[i];
  6. return ref; // 返回第 i 个元素的引用,ref 是一个引用变量,ref 引用 vals[i],最后再返回 shit。
  7. }
  8. int main ()
  9. {
  10. cout << "改变前的值" << endl;
  11. cout << "vals[0]: " << vals[0] <<endl;
  12. cout << "&vals[0]: " << &vals[0] <<endl;
  13. for ( int i = 0; i < 4; i++ )
  14. {
  15. cout << "vals[" << i << "] = " << vals[i] << endl;;
  16. }
  17. cout << "setValues(0): " << setValues(0) <<endl;
  18. cout << "&setValues(0): " << &setValues(0) <<endl;
  19. setValues(0) = 11.11; // 改变第 1 个元素
  20. setValues(1) = 22.22; // 改变第 2 个元素
  21. setValues(2) = 33.33; // 改变第 2 个元素
  22. setValues(3) = 44.44; // 改变第 4 个元素
  23. cout << "改变后的值" << endl;
  24. for ( int i = 0; i < 4; i++ )
  25. {
  26. cout << "vals[" << i << "] = " << vals[i] << endl;;
  27. }
  28. return 0;
  29. }
  30. /*
  31. 改变前的值
  32. vals[0]: 1.1
  33. &vals[0]: 0x102eb1100
  34. vals[0] = 1.1
  35. vals[1] = 2.2
  36. vals[2] = 3.3
  37. vals[3] = 0
  38. setValues(0): 1.1
  39. &setValues(0): 0x102eb1100
  40. 改变后的值
  41. vals[0] = 11.11
  42. vals[1] = 22.22
  43. vals[2] = 33.33
  44. vals[3] = 44.44
  45. */

当返回一个引用时,要注意被引用的对象不能超出作用域。所以返回一个对局部变量的引用是不合法的,但是,可以返回一个对静态变量的引用。

  1. int& func() {
  2. int q;
  3. //! return q; // ❌ 在编译时发生错误
  4. static int x;
  5. return x; // ✅ 安全,x 在函数作用域外依然是有效的
  6. }