[C++] C++入门第二篇 -- 引用& -- 内联函数inline -- auto+for(下)

简介: [C++] C++入门第二篇 -- 引用& -- 内联函数inline -- auto+for(下)

2、内联函数

2.1 概念

以inline修饰的函数叫做内联函数,编译时C++编译器会在调用内联函数的地方展开,没有函数调用建立栈帧的开销,内联函数提升程序运行的效率。

我们来看一下平常我们写的代码:

int Add(int x, int y)
{
  return x + y;
}
int main()
{
  int ret = 0;
  ret = Add(1, 2);
  return 0;
}

我们可以看到,这里是在调用函数,但是我们要是不断要用Add函数的时候,不断的调用效率会比较低,因此在C++中,我们引入了内联函数(inline)。

inline int Add(int x, int y)
{
  return x + y;
}
int main()
{
  int ret = 0;
  ret = Add(1, 2);
  return 0;
}


我们可以看到,加了inline变为内联函数后,就不再是调用了,直接用函数体替换了函数调用,不用开栈帧,可以提高效率。


看到这是不是想到,C++的内联函数像是C语言的宏。


C++中的内联函数确实和C语言的宏用途是一样的,对于短小且频繁调用的函数,C语言用宏来代替函数,C++中用内联函数。


C++是全面兼容C语言的,我们直接用宏就可以了,那为什么我们还要使用内联函数呢?


1、宏在写的时候容易出错,且没有类型的检查,还不能调试。


2、内联函数会对参数的类型进行检查,还可以调试,书写上就是正常的写功能函数,在返回值类型前加inline。


如果想要看底层是调用还是直接展开的,查看方式:

1. 在release模式下,查看编译器生成的汇编代码中是否存在call Add

2. 在debug模式下,需要对编译器进行设置,否则不会展开(因为debug模式下,编译器默认不会对代码进行优化,以下给出vs2019的设置方式)





2.2 特性

1. inline是一种以空间换时间的做法,如果编译器将函数当成内联函数处理,在编译阶段,会用函数体替换函数调用,缺陷:可能会使目标文件变大,优势:少了调用开销,提高程序运行效率。

2. inline对于编译器而言只是一个建议,编译器会自动优化,如果内联函数内存在循环/递归的时候,编译器会自动优化忽略掉内联。(一般建议10行以内)


3. inline不建议声明和定义分离,分离会导致链接错误。因为inline直接再调用处被展开,(不会出现在符号表中)就没有函数地址了,链接就会找不到。


对于第二点我们做一下实验:

inline int Add(int x, int y)
{
  int sum = x + y;
  sum += x * y;
  sum += x * y;
  sum += x * y;
  sum += x * y;
  sum += x * y;
  sum += x * y;
  sum += x * y;
  sum += x * y;
  sum += x * y;
  return sum;
}
int main()
{
  int ret = 0;
  ret = Add(1, 2);
  return 0;
}


这里内联函数函数体一共写了十一行就算是函数调用了。

3、auto

3.1 auto简介

在早期C/C++中auto的含义是:使用auto修饰的变量,是具有自动存储器的局部变量。C++11中,标准委员会赋予了auto全新的含义即:auto不再是一个存储类型指示符,而是作为一个新的类型指示符来指示编译器,auto声明的变量必须由编译器在编译时期推导而得。


auto我们在C语言期间就接触过,C语言期间定义的局部变量默认是用auto修饰,因此我们在定义变量的时候从来不加auto,也就没人在意。但是到了C++11时期,auto有了新的身份,它可以自动推导类型。


我们来看一段代码,看看auto的自动推导类型:

int testAuto()
{
  return 1;
}
int main()
{
  int a = 0;
  auto b = a;
  auto c = 'c';
  auto ret = testAuto();
  cout << typeid(b).name() << endl;
  cout << typeid(c).name() << endl;
  cout << typeid(ret).name() << endl;
  return 0;
}

运行结果:

这段代码里面 typeid(变量名).name() 是推导变量类型的一个函数。


我们可以看到auto很智能,可以根据赋的值来推导类型。


注意:使用auto定义变量时必须对其进行初始化,在编译阶段编译器需要根据初始化表达式来推导auto的实际类型。因此auto并非是一种“类型”的声明,而是一个类型声明时的“占位符”,编译器在编译期会将auto替换为变量实际的类型。

3.2 auto的使用细则

1. auto与指针和引用结合起来使用:用auto声明指针类型时,用auto和auto*没有任何区别,但用auto声明引用类型时则必须加&。

int main()
{
  int x = 10;
  auto a = &x;
  auto* b = &x;
  auto& c = x;
  cout << typeid(a).name() << endl;
  cout << typeid(b).name() << endl;
  cout << typeid(c).name() << endl;
  return 0;
}

运行结果:

2. 在同一行定义多个变量:当在同一行声明多个变量时,这些变量必须是相同的类型,否则编译器将会报错,因为编译器实际只对第一个类型进行推导,然后用推导出来的类型定义其他变量。

void TestAuto()
{
    auto a = 1, b = 2;
    auto c = 3, d = 4.0; // 该行代码会编译失败,因为c和d的初始化表达式类型不同
}



3.3 auto不能推导的场景

1. auto不能作为函数的参数

// 此处代码编译失败,auto不能作为形参类型,因为编译器无法对a的实际类型进行推导
void TestAuto(auto a)
{}


2. auto不能直接用来声明数组

void TestAuto()
{
    int a[] = {1,2,3};
    auto b[] = {4,5,6};
}



3.4 auto与for合用

按照C语言我们的写法,遍历数组是下面的代码

int main()
{
  int array[] = { 1, 2, 3, 4, 5 };
  for (int i = 0; i < sizeof(array) / sizeof(array[0]); ++i)
    array[i] *= 2;
  for (int i = 0; i < sizeof(array) / sizeof(array[0]); ++i)
    cout << array[i] << " ";
  cout << endl;
  return 0;
}

运行结果:

我们现在也可以使用auto这样来遍历数组:

int main()
{
  int array[] = { 1, 2, 3, 4, 5 };
  for (auto e : array)
    cout << e << " ";
  return 0;
}

运行结果:

我们这里使用的是范围 for ,for循环后的括号由冒号”:“分为两个部分:第一部分是范围内用于迭代的变量,第二部分表示迭代的范围。这里会自动判断结束的。


这里的e是取到数组里的元素,然后打印,不会影响数组元素。


如果想改变数组元素,我们可以使用auto& e,这是对数组元素起别名,直接改变数组元素,auto取到元素后会自动推导类型的。如下:

int main()
{
  int array[] = { 1, 2, 3, 4, 5 };
  for (auto& e : array)
    e *= 2;
  for (auto e : array)
    cout << e << " ";
  return 0;
}

运行结果:

我们可以看到结果,这样写就把数组元素改了。

相关文章
|
1月前
|
安全 编译器 程序员
【C++初阶】C++简单入门
【C++初阶】C++简单入门
|
13天前
|
编译器 Linux C语言
C++基础入门
C++基础入门
|
13天前
|
C语言 C++
C++(三)内联函数
本文介绍了C++中的内联函数概念及其与宏函数的区别。通过对比宏函数和普通函数,展示了内联函数在提高程序执行效率方面的优势。同时,详细解释了如何在C++中声明内联函数以及其适用场景,并给出了示例代码。内联函数能够减少函数调用开销,但在使用时需谨慎评估其对代码体积的影响。
|
1月前
|
安全 编译器 C语言
C++入门-数组
C++入门-数组
|
1月前
|
存储 编译器 程序员
C++从遗忘到入门
本文主要面向的是曾经学过、了解过C++的同学,旨在帮助这些同学唤醒C++的记忆,提升下自身的技术储备。如果之前完全没接触过C++,也可以整体了解下这门语言。
|
1月前
|
存储 编译器 C++
【C++关键字】auto的使用(C++11)
【C++关键字】auto的使用(C++11)
|
1月前
|
C++ 容器
C++中自定义结构体或类作为关联容器的键
C++中自定义结构体或类作为关联容器的键
31 0
|
13天前
|
存储 编译器 C++
C ++初阶:类和对象(中)
C ++初阶:类和对象(中)
|
13天前
|
C++
C++(十六)类之间转化
在C++中,类之间的转换可以通过转换构造函数和操作符函数实现。转换构造函数是一种单参数构造函数,用于将其他类型转换为本类类型。为了防止不必要的隐式转换,可以使用`explicit`关键字来禁止这种自动转换。此外,还可以通过定义`operator`函数来进行类型转换,该函数无参数且无返回值。下面展示了如何使用这两种方式实现自定义类型的相互转换,并通过示例代码说明了`explicit`关键字的作用。
|
13天前
|
存储 设计模式 编译器
C++(十三) 类的扩展
本文详细介绍了C++中类的各种扩展特性,包括类成员存储、`sizeof`操作符的应用、类成员函数的存储方式及其背后的`this`指针机制。此外,还探讨了`const`修饰符在成员变量和函数中的作用,以及如何通过`static`关键字实现类中的资源共享。文章还介绍了单例模式的设计思路,并讨论了指向类成员(数据成员和函数成员)的指针的使用方法。最后,还讲解了指向静态成员的指针的相关概念和应用示例。通过这些内容,帮助读者更好地理解和掌握C++面向对象编程的核心概念和技术细节。