【C++精华铺】3.C++入门 引用(const)、内联函数

简介: 以值作为参数或者返回类型,在传参和返回的期间,变量不会直接传递实参或返回变量本身,而传递的是实参或者变量的一份临时拷贝,如果需要拷贝的类型特别大的时候,效率是极低的。常引用就是const关键字修饰的引用,但是我们在使用常引用是会遇到各种报错,其实就是权限放大缩小的问题,所谓权限就是是否具有读或者写的权力,在C++中引用支持权限缩小但不支持权限放大。,与传统的定义变量不同,传统的定义变量是新开一份空间来存储数据,而引用则是给一个已经存在的空间起一个别名,它和引用的变量共用同一份空间。

 目录

1. 引用

1.1 引用特性

1.2 常引用

1.2.1 权限放大

1.2.2 权限缩小

1.3 使用场景

1.3.1 传参

1.3.2 做返回值

1.4 传值和传引用的效率比较

1.5 引用和指针的区别

2. 内联函数

2.1 inline

2.2 特性


1. 引用

       在C++中,引入了一个新的概念引用,与传统的定义变量不同,传统的定义变量是新开一份空间来存储数据,而引用则是给一个已经存在的空间起一个别名,它和引用的变量共用同一份空间。

image.gif编辑

1.1 引用特性

    • 引用必须在定义的时候初始化
    • 一个变量可以有多个引用
    • 引用一旦引用一个实体就不能在引用其他实体
    int a = 0;
      int& b = a;
      int& c = a;
      cout << a << ' ' << b << ' ' << c << endl;
      cout << &a << ' ' << &b << ' ' << &c << endl;

    image.gif

    输出:

    0 0 0

    000000B0DDAFF6A4 000000B0DDAFF6A4 000000B0DDAFF6A4

           从上述的输出可以很明显的看出引用是和被引用的实体共用用一块空间。

    1.2 常引用

           常引用就是const关键字修饰的引用,但是我们在使用常引用是会遇到各种报错,其实就是权限放大缩小的问题,所谓权限就是是否具有读或者写的权力,在C++中引用支持权限缩小但不支持权限放大。

    1.2.1 权限放大

           首先要说明的是c++是不支持权限放大的,如果出现权限放大就会出现报错。(权限放大就是增加了权限,比如原本变量用const修饰只能读不能修改,你用引用引用它时没有使用const修饰即为可以修改,这就叫做权限放大,这是不被编译器允许的。)

    const int a = 0;
      //错误写法
      int& ra = a; //变量a有const修饰符,说明a只读不可写入,这里没有用const修饰,造成权限放大。
      //正确写法
      const int& rra = a;

    image.gif

           还有一种情况特别容易发生错误,而且犯错的人常常觉得自己没有发生权限放大,但是确确实实发生了,并且发生了报错,这就是会发生类型转换的场景。比如下面的场景。

    image.gif编辑

    1.2.2 权限缩小

           在C++中权限缩小是被允许的,权限缩小就是比如原来的变量可读可写,你用一个const引用引用这个实体是完全可以的,比如:

    int a = 0;
      const int& b = a;

    image.gif

    1.3 使用场景

    1.3.1 传参

           传参的一个特别典型的应用就是swap了,因为引用所引用的实体和本体共用的是一个空间,所以引用发生改变其实也就是本体发生改变。所以swap交换变量值就是一个很典型的应用。

    void mySwap(int& a, int& b)
    {
      swap(a, b);
    }
    int main()
    {
      int a = 0, b = 1;
      cout << a << b << endl;
      swap(a, b);
      cout << a << b << endl;
    }

    image.gif

    输出:

    01

    10

    1.3.2 做返回值

    int& count()
    {
      static int a = 0;
      a++;
      //...
      return a;
    }

    image.gif

            但是我们在返回引用的时候要特别注意被引用对象的作用域和声明周期,以免导致非凡的访问,比如下面的案例:

    //错误案例
    int& add(int a, int b)
    {
      int c = a + b;
      return c;
    }
    int main()
    {
      cout << add(1, 2) << endl;
    }

    image.gif

    image.gif编辑

           所以这里纵使编译器返回了正确的结果甚至也没有报错,但它依然是一种错误的使用。

    1.4 传值和传引用的效率比较

           以值作为参数或者返回类型,在传参和返回的期间,变量不会直接传递实参或返回变量本身,而传递的是实参或者变量的一份临时拷贝,如果需要拷贝的类型特别大的时候,效率是极低的。而传引用是直接传递实参或返回变量的本身,几乎没有什么消耗。    image.gif编辑

    1.5 引用和指针的区别

           在语法概念上引用其实是一个别名没有独立的空间,和其引用的实体共用同一块空间。但是在底层的实现上实际上是有空间的,因为引用是按照指针方式来实现的。我们通过引用和指针的汇编代码就可以很容易的看出来。image.gif编辑

           引用和指针的不同点:

      1. 引用在概念上定义一个变量的别名,指针存储一个变量地址。
      2. 引用在定义的时候必须初始化,指针没有要求。
      3. 引用只能引用一个实体,不能更改,指针可以在任何时候指向一个任意的同类型的实体。
      4. 没有NULL引用,有NULL指针。
      5. sizeof(引用)是引用类型的大小,但指针始终是地址空间所占字节个数。
      6. 引用自增为所引用的实体自增,指针自增表示向后偏移一个类型的大小。
      7. 指针需要显式解引用,引用编译器自己处理。
      8. 引用比指针安全。

      2. 内联函数

      2.1 inline

             以inline关键字修饰的函数叫做内联函数,编译时C++会在调用内联函数的地方展开,没有函数建立栈帧的开销,提升程序运行的效率。C++期望用inline去替代c语言的宏(因为宏不方便调试,会使代码可读性降低,可维护性差,没有类型的安全检查)

      image.gif编辑

      2.2 特性

             inline是一种空间换时间的做法,如果编译器将函数当成内联函数处理,在编译阶段,会用将函数体替换函数调用,但会让目标文件变大。

             inline对于编译器只是一个建议,不同的编译器关于inline的实现机制有可能不同,一般将函数规模较小的函数不是递归且频繁调用的函数采用inline修饰,否则编译器可能会忽略inline特性。

             inline也不支持声明和定义分离分离会导致链接错误,因为inline函数会被展开,没有函数地址,如果声明和定义分开会导致链接阶段找不到函数的地址从而报错。image.gif编辑


      相关文章
      |
      1月前
      |
      存储 安全 编译器
      第二问:C++中const用法详解
      `const` 是 C++ 中用于定义常量的关键字,主要作用是防止值被修改。它可以修饰变量、指针、函数参数、返回值、类成员等,确保数据的不可变性。`const` 的常见用法包括:
      108 0
      |
      3月前
      |
      编译器 C++
      C++入门12——详解多态1
      C++入门12——详解多态1
      59 2
      C++入门12——详解多态1
      |
      3月前
      |
      C++
      C++入门13——详解多态2
      C++入门13——详解多态2
      96 1
      |
      3月前
      |
      存储 安全 编译器
      【C++打怪之路Lv1】-- 入门二级
      【C++打怪之路Lv1】-- 入门二级
      39 0
      |
      3月前
      |
      自然语言处理 编译器 C语言
      【C++打怪之路Lv1】-- C++开篇(入门)
      【C++打怪之路Lv1】-- C++开篇(入门)
      44 0
      |
      3月前
      |
      分布式计算 Java 编译器
      【C++入门(下)】—— 我与C++的不解之缘(二)
      【C++入门(下)】—— 我与C++的不解之缘(二)
      |
      3月前
      |
      编译器 Linux C语言
      【C++入门(上)】—— 我与C++的不解之缘(一)
      【C++入门(上)】—— 我与C++的不解之缘(一)
      |
      3月前
      |
      编译器 C++
      C++入门11——详解C++继承(菱形继承与虚拟继承)-2
      C++入门11——详解C++继承(菱形继承与虚拟继承)-2
      49 0
      |
      11天前
      |
      C++ 芯片
      【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
      声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
      51 18
      |
      11天前
      |
      存储 编译器 数据安全/隐私保护
      【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
      声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
      37 13