c++学习笔记(二):引用

c++中的引用

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


C++ 引用 vs 指针

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

  • 不存在空引用。引用必须连接到一块合法的内存。

  • 一旦引用被初始化为一个对象,就不能被指向到另一个对象。指针可以在任何时候指向到另一个对象。

  • 引用必须在创建时被初始化。指针可以在任何时间被初始化。

    ​ int &b; //错误的 int *p; //正确的

引用的基本使用

作用:给变量起别名

语法:数据类型 &别名 = 原名

int main()
{
	int a = 10;
    int &b = a;
    
    std::cout << a << std::endl;
    std::cout << b << std::endl;
    //a = 10; b = 10;
    b = 100;
    std::cout << a << std::endl;
    std::cout << b << std::endl;
    //a = 100; b = 100;
    return 0;
}

引用用于函数参数列表和函数返回值。

引用做函数参数

作用:函数传参时,可以利用引用的技术让形参修饰(修饰)实参

优点:可以简化指针修改实参

void swap01(int a, int b) //值传递
{
    int temp = a;
    a = b;
    b = a;
}
void swap02(int* a, int* b) //指针(地址)传递
{
    int temp = *a;
    *a = *b;
    *b = *a;
}
void swap03(int& a, int& b) //引用传递
{
    int temp = &a;
    &a = &b;
    &b = &a;
}
int main()
{
    int a = 1, b = 2, c = 3, d = 4, i = 5, j = 6;
    swap01(a, b);//值传递形参不改变(修饰)实参
    swap02(&c, &d);//指针传递将地址传递过去,可以改变(修饰)实参
    swap03(i, j);//同样可以改变(修饰)实参,但相比指针传递更间接方便
}

引用做函数返回值

注意:不要返回局部变量引用

用法:函数调用可以作为左值

C++ 函数可以返回一个引用,方式与返回一个指针类似。

当函数返回一个引用时,则返回一个指向返回值的隐式指针。这样,函数就可以放在赋值语句的左边。

double vals[] = {10.1, 12.6, 33.1, 24.1, 50.0};
 
double& setValues(int i) //函数名前加&代表该函数返回引用 
{  
   double& ref = vals[i];    
   return ref;   // 返回第 i 个元素的引用,ref 是一个引用变量,ref 引用 vals[i]
}
 
// 要调用上面定义函数的主函数
int main ()
{
 
   cout << "改变前的值" << endl;
   for ( int i = 0; i < 5; i++ )
   {
       cout << "vals[" << i << "] = ";
       cout << vals[i] << endl;
   }
 /*
改变前的值
vals[0] = 10.1
vals[1] = 12.6
vals[2] = 33.1
vals[3] = 24.1
vals[4] = 50
*/
   setValues(1) = 20.23; // 改变第 2 个元素
   setValues(3) = 70.8;  // 改变第 4 个元素
 
   cout << "改变后的值" << endl;
   for ( int i = 0; i < 5; i++ )
   {
       cout << "vals[" << i << "] = ";
       cout << vals[i] << endl;
   }
/*
改变后的值
vals[0] = 10.1
vals[1] = 20.23
vals[2] = 33.1
vals[3] = 70.8
vals[4] = 50
*/
   return 0;
}

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

int& func() {
   int q;
   //! return q; // 在编译时发生错误
   static int x;
   return x;     // 安全,x 在函数作用域外依然是有效的
}
#include <iostream>
using namespace std;

// 返回对静态变量的引用
int& getStaticRef() {
    static int num = 5; // 静态变量
    return num;
}

int main() {
    int& ref = getStaticRef(); // 获取对静态变量的引用
    cout << "初始值:" << ref << endl;// 5

    ref = 10; // 修改静态变量的值

    cout << "修改后的值:" << ref << endl;// 10
    cout << "再次调用函数后的值:" << getStaticRef() << endl; // 10

    return 0;
}

引用的本质

引用的本质在c++内部实现是一个指针常量

//发现是引用,转换为 int* const ref = &a;
void func(int& ref)
{
    ref = 100; //ref是引用,转换为*ref = 100
}
int main()
{
    int a =10;
    
    //自动转换为 int* const ref = &a;指针常量是指向不可改的指针,也说明了为什么引用不可更改
    int& ref = a;
    ref = 20; //内部发现ref是引用,自动帮我们转换为 *ref = 20;
    
    std::cout<< "a:" << a <<std::endl;
    std::cout<< "ref:" << ref <<std::endl;
    
    func(a);
    return 0;
}

常量引用

作用:常量引用主要用来修饰形参,防止误操作

在函数形参列表中,可以加const修饰形参,防止形参改变实参

//引用使用的场景,通常原来修饰形参
void showValue(const int& v)
{
    //v += 10; 加入const后报错,防止形参改变实参
    std::cout<< v <<std::endl;
}

int main()
{
    //int& ref = 10; 引用必须引用一块合法的内存空间,此处直接引用了作为常量的10
    //const int& ref = 10; 在int前加上const之后为合法,因为编译器将代码修改为:int temp = 10; const int& ref = temp;
    //ref = 20; 加入const后为只读状态
    
    int a = 100;
    showValue(a);
    
    return 0;
}
posted @ 2024-06-16 17:08  StaDark  阅读(13)  评论(0编辑  收藏  举报