C语言到C++的跨越(3)
内联函数
内联函数概念
以inline修饰的函数叫做内联函数,编译时C++编译器会在调用内联函数的地方展开,没有函数调用建立栈帧的开销,内联函数提升程序运行的效率。
宏函数与内联函数的对比
宏函数不对函数进行展开,会建立函数栈帧,不方便调试
int Add(int x, int y)
{
return x + y;
}
int main()
{
int ret = 0;
ret = Add(1, 2);
return 0;
}
而使用内敛函数
inline int Add(int x, int y)
{
return x + y;
}
int main()
{
int ret = Add(1, 2);
cout << ret << endl;
return 0;
}
这里可以看出,没有函数的地址,而是直接展开函数,方便观察与调用,而且还很方便,不容易出错。
但是当函数的语句过长(行数太长),就不会展开了。因为展开后可执行程序变大,函数规模较小的情况下才可以使用。
// F.h
#include <iostream>
using namespace std;
inline void f(int i);
// F.cpp
#include "F.h"
void f(int i)
{
cout << i << endl;
}
// main.cpp
#include "F.h"
int main()
{
f(10);
return 0;
}
// 链接错误:main.obj : error LNK2019: 无法解析的外部符号 "void __cdeclf(int)" (?f@@YAXH@Z),该符号在函数 _main 中被引用
为什么定义和声明分离会报链接错误呢?
函数重载里面讲过,链接要找函数的地址,而内敛函数中没有函数的地址,因为直接把函数展开了,所以使用内敛函数时,声明和定义要放在一起。
auto关键字(初步认识)
auto可以自动识别变量类型。
使用auto定义变量时必须对其进行初始化
auto不能作为形参类型,因为编译器无法对a的实际类型进行推导
基于范围的for循环
对于一个有范围的集合而言,由程序员来说明循环的范围是多余的,有时候还会容易犯错误。因此C++11中引入了基于范围的for循环。
void TestFor()
{
int array[] = { 1, 2, 3, 4, 5 };
for(auto& e : array)
e *= 2;
for(auto e : array)
cout << e << " ";
return 0;
}
注意,这种for循环只适合于从前到后,如果要从后到前则要像c语言中一样写。
for循环迭代的范围必须是确定的
对于数组而言,就是数组中第一个元素和最后一个元素的范围;对于类而言,应该提供begin和end的方法,begin和end就是for循环迭代的范围。
void TestFor(int array[])
{
for(auto& e : array)
cout<< e <<endl;
}
由于这里不知道参数范围、大小,所以不可以这样使用。
指针空值nullptr
为什么NULL用的好好的,要换nullptr?
本来想通过f(NULL)调用指针版本的f(int*)函数,可是事与愿违。
NULL可能被定义为字面常量0,或者被定义为无类型指针(void*)的常量。
所以为了提高代码的健壮性,在后续表示指针空值时建议最好使用nullptr。