文章目录
前言一.引用1.引用的概念2.引用的特性3.引用的使用场景4.常引用5.引用和指针的区别 二.内联函数1.C语言的宏函数2.内联函数的概念3.内联函数的特性 三.auto关键字1.`auto`的定义2.`auto`的使用规则3.`auto`不能推导的场景 四.基于范围的for循环1.范围for的语法2.范围for的使用条件 五.指针空值nullptr
前言
在上一篇文章中讲解了部分c++入门知识点,这篇文章将继续讲解剩下的知识点
一.引用
1.引用的概念
**引用(&
)**不是新创建一个变量,而是为变量取别名。引用时共用一块空间,编译器不会新开辟空间。如图所示,b是a的别名,c是b的别名,它们的空间地址一样
using namespace std;int main() {int a = 1;int& b = a;int& c = b;cout << &a << endl;cout << &b << endl;cout << &c << endl;return 0;}
C语言中用指针对变量进行修改,而c++可以直接用引用进行修改。
using namespace std;int main() {int a = 1;int& b = a;int& c = b;cout << a << " ";cout << b << " ";cout << c << " ";cout << endl;b++;cout << a << " ";cout << b << " ";cout << c << " ";cout << endl;c++;cout << a << " ";cout << b << " ";cout << c << " ";cout << endl;return 0;}
2.引用的特性
引用时必须初始化
int a=0;int &b; //错误int &b=a; //正确
一个变量可以有多个引用
int a=0;int &b=a;int &c=a;
引用一旦引用一个实体,再不能引用其他实体
int a=0;int c=0;int &b=a;int &b=c; //错误
3.引用的使用场景
做参数
我们之前用C语言写交换两个数的函数时需要用到指针传参,而c++中可以直接用引用做参数。比如下面的代码中,形参a,b就是实参x,y的别名(也就是引用),形参a,b交换,实参x,y也会交换。引用做参数可以提高函数调用的效率。
using namespace std;//用指针做参数void Swap1(int*a,int*b){ int t=*a; *a=*b; *b=t;}//用引用做参数void Swap2(int&a,int&b){ int t=a; a=b; b=t;}int main(){ int x=10,y=20; Swap1(&x,&y); cout<<x<<" "<<y<<endl; x=10,y=20; Swap2(x,y); cout<<x<<" "<<y<<endl; return 0;}
做返回值
函数传值返回时,会生成一个临时变量用来拷贝返回值,而引用做返回值时,不用生成临时变量,减少了拷贝,提高了效率。
int& Count(){ static int n=0; n++; return n;}int main(){ int ret=Count(); return 0;}
上面这段代码,返回的n在静态区,所以函数调用结束时,n依然还在。而下面这段代码则是引用做返回值时的错误用例:
int& Count(){ int n=0; n++; return n;}int main(){ int ret=Count(); return 0;}
不同的是下面的n是局部变量,当函数调用结束时,栈帧销毁,如果没有清理栈帧,ret可能是正确值,如果清理栈帧,ret就会是一个随机值。而如果接受也是引用时,那么ret就一定是随机值。比如:
int& Count(){ int n=0; n++; return n;}int main(){ //ret是n的别名 int& ret=Count(); return 0;}
总结:
基本任何场景都可以用引用传参。谨慎使用引用做返回值,出了函数作用域,对象不在就不能用引用返回,还在就可以用引用返回。4.常引用
int main(){ const int a=0; int& b=a; //不可以,权限不能放大 const int c=0; int d=c; //可以,c拷贝给d,权限平移,d的改变不影响c int x=0; const int& y=x;//可以,权限缩小,如果x改变,y也会改变,但不能通过y来修改x double dd=1.11; int ii=dd; //可以,借助int临时变量 const int& rii=dd;//不可以,临时变量具有常性}
5.引用和指针的区别
引用概念上定义一个变量的别名,指针存储一个变量地址引用在定义时必须初始化,指针没有要求引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以在任何时候指向任何 一个同类型实体没有NULL引用,但有NULL指针在sizeof中含义不同:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32 位平台下占4个字节)引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小有多级指针,但是没有多级引用访问实体方式不同,指针需要显式解引用,引用编译器自己处理引用比指针使用起来相对更安全二.内联函数
1.C语言的宏函数
我们首先来看下面这段代码:
int Add(int x,int y){ return x+y;}int main(){ for(int i=0;i<10000;i++){ cout<<Add(i,i+1)<<endl; } return 0;}
在调用函数时,每调用一次函数,就要在栈区创建和销毁一次空间,而我们上面这一段代码,频繁地调用函数,就会频繁地建立栈帧,大大降低了效率,为了解决这一情况,C语言用宏函数来替换:
#define Add(x,y) ((x)+(y))
利用宏函数来替换,不需要建立栈帧,大大提高调用效率。但缺点就是写起来较为复杂,容易出错,可读性差并且不能调试。
而在c++中为优化这一点,增加了新的内联函数inline
。
2.内联函数的概念
以inline
修饰的函数叫做内联函数,编译时c++编译器会在调用内联函数的地方展开,没用调用函数建立栈帧,提升程序运行的效率。
还是上面的这一段代码:
//在函数类型前加上inline关键字inline int Add(int x,int y){ return x+y;}int main(){ for(int i=0;i<10000;i++){ cout<<Add(i,i+1)<<endl; } return 0;}
3.内联函数的特性
inline
是一种以空间换时间的做法,编译器将函数当成内联函数处理时,在编译阶段,会展开整个函数体来替换函数调用。少了调用开销,大大提高程序运行效率。内联函数只适合函数规模较小的(也就是函数不是很长),不是递归且频繁调用的函数用inline
修饰。比如函数fun()
有50行代码,如果fun()
不是内联函数时,10000
个位置调用函数,合计共10000+50
行,如果fun()
是内联函数,合计共10000*50
行。这就会使目标文件变大。内联函数不建议声明和定义分离,分离会导致链接错误,因为inline
被展开,就没有函数地址,链接找不到。 //fun.h#include<iostream>using namespace std;inline void fun(int i);//fun.cpp#include"fun.h"void fun(int i) {cout << i << endl;}//test.cpp#include"fun.h"int main() {int x = 10;fun(10);return 0;}
最好声明和定义放在一起:
//fun.h#include<iostream>using namespace std;void fun(int i) {cout << i << endl;}//test.cpp#include"fun.h"int main() {int x = 10;fun(10);return 0;}
三.auto关键字
1.auto
的定义
在C语言的时候我们知道typedef
关键字可以用来给一些类型取别名,对于一些较长的类型名时使用起来会很方便,比如,std::map<std::string,std::string>::iterator
是一个类型,但是该类型太长了,特别容易写错,于是可以通过typedef给类型取别名
typedef std::map<std::string, std::string> Map;
使用typedef
确实可以简便代码,但是也会遇到一些特殊情况:
typedef char*pstringint main(){ const pstring p1; const pstring *p2; return 0;}
c++为了解决这一情况将auto
赋予了新的含义。
c++11中,auto
不再是一个存储类型指示符,而是作为一个新的类型指示符来指示编译器,auto
声明的变量必须由编译器在编译时期推导而得。
int TestAuto(){return 10;}int main(){int a = 10;auto b = a;auto c = 'a';auto d = TestAuto();cout << typeid(b).name() << endl;cout << typeid(c).name() << endl;cout << typeid(d).name() << endl;return 0;}
2.auto
的使用规则
使用auto
时,一定要初始化,在编译阶段编译器需要根据初始化表达式来推到实际类型,然后将auto
替换为变量的实际类型。
auto
声明指针类型时,auto
和auto*
没有区别,但是auto
声明引用类型时必须加上&
。
int main() {int a = 10;auto b = &a;auto* c = &a;auto& d = a;cout << typeid(b).name() << endl;cout << typeid(c).name() << endl;cout << typeid(d).name() << endl;*b = 20;*c = 30;d = 40;return 0;}
在同一行定义多个变量时,这些变量必须是相同的类型,否则编译器会报错。
void Testauto(){ auto a=1,b=2; auto c=1,d=1.1;}
3.auto
不能推导的场景
auto
不能作为函数的参数
void TestAuto(auto a){ cout<<a<<endl;}
**auto
不能直接用来声明数组**
void TestAuto(){ int a[]={1,2,3}; auto b[]={4,5,6};}
四.基于范围的for循环
1.范围for的语法
在C语言或者c++98中,我们如果要遍历一个数组,通常会按照以下方式进行
using namespace std;int main() { int array[] = { 1,2,3,4,5,6 }; for (int i = 0; i < sizeof(array) / sizeof(int); i++) { array[i] *= 2; } for (int i = 0; i < sizeof(array) / sizeof(int); i++) { cout << array[i] << " "; } cout << endl; return 0;}
而在之后的c++11更新中,引入了基于范围的for循环,对于一个有范围的集合,在遍历时可以自动识别循环范围。for循环后的括号由:
分为两部分,第一部分是范围内用于迭代的变量,第二部分则表示被迭代的范围。
using namespace std;int main() { int array[] = { 1,2,3,4,5,6 }; for (auto& e : array) { //每一项都乘以2 e *= 2; } for (auto& e : array) { cout << e << " "; } cout << endl; return 0;}
和普通的for循环一样,范围for循环也可以用continue结束本次循环,也可以用break直接结束整个循环。
2.范围for的使用条件
for循环迭代的范围必须是确定的。
对于一个数组,就是数组第一个元素和最后一个元素的范围;在之后学到类时,还有关于类的范围。
下面这一段代码就是错误的,因为for的范围不确定,数组名作为参数传过来,我们只知道数组第一个元素,而不确定最后一个。
using namespace std;void TestFor(int array[]){ for(auto e:array){ cout<<e<<endl; }}int main(){ int array[]={1,2,3,4,5,6}; TestFor(array); return 0;}
五.指针空值nullptr
在C语言的时候我们知道NULL
表示空指针,但实际上NULL
是一个宏,NULL
被定义为字面常量0或者无类型指针(void*
)的常量。在C语言的头文件(stddef.h)中可以看到以下代码:
#ifndef NULL#ifdef _cplusplus#define NULL 0#else#define NULL ((void*)0)#endif#endif
在使用NULL
不可避免的会遇到一些特殊情况,比如:
void f(int) { cout << "f(int)" << endl;}void f(int*) { cout << "f(int*)" << endl;}int main() { f(0); f(NULL); return 0;}
因为NULL
被定义为0,因此程序默认调用了第一个函数void f(int)
,如果要使NULL
按照指针方式使用,必须1强制转换为`((void*)0)。
为了解决这个麻烦,在c++11时,引入了新关键字nullptr
表示指针空值,在使用时,不需要包含头文件。
在上面的代码中加上这句,就会得到以下结果:
f(nullptr);
在c++11中,sizeof(nulllptr)
与sizeof((void*)0)
所占的字节数相同。
以上就是关于c++入门部分的讲解,如果哪里有错的话,可以在评论区指正,也欢迎大家一起讨论学习,如果对你的学习有帮助的话,点点赞关注支持一下吧!!!