C++类与对象深入之引用与内联函数与auto关键字及for循环详解

目录
  • 一:引用
    • 1.1:概念
    • 1.2:引用特性
    • 1.3:常引用
    • 1.4:使用场景
    • 1.5:引用和指针的区别
  • 二:内联函数
    • 2.1:概念
    • 2.2:特性
    • 2.3:面试题
  • 三:auto关键字
    • 3.1:auto简介
    • 3.2:auto使用细则
    • 3.3:auto不能推导的场景
  • 四:基于范围的for循环
    • 4.1:范围for循环的语法
    • 4.2:范围for循环的使用条件

一:引用

1.1:概念

引用不是定义一个新的变量,而是给已经存在的变量取一个别名。注意:编译器不会给引用变量开辟内存空间,他和他的引用变量共用同一块内存空间。

类型& 引用变量名(对象名) = 引用实体。

1.2:引用特性

1. 引用在定义时必须初始化

2. 一个变量可以有多个引用

3. 引用一旦引用一个实体,也就不能引用其他实体

通俗的讲就是:我们取外号,肯定是对一个对象取外号,不可能是这空气取外号,而且也可以给同一个人取多个外号,但是同一个外号我们就不要给多个人取了,那样就乱套了,就不知道叫的是谁了。

1.3:常引用

原则:对原引用变量,权限只能缩小,不能放大。

int main(){
		const int x = 20;
		//int& y = x;       // 放大
		const int& y = x;   // 不变
		int c = 30;
		const int& d = c;  // 缩小
		//cout << d << endl;
	system("pause");
	return 0;
}

对常变量取别名时,要加const;

const int & b = 10;

注意:当引用类型和引用实体不是同一类型时,如:

double a = 3.14;
//int &ra = a; //该编译语句会报错,因为类型不同
const int & ra = a;
//加上const就可以了。

我们可以这么理解,当浮点型转换整型数据时,其中我们在C语言学过会发生隐式类型转换,在转换的时候会产生一个临时变量,这个临时变量具有常性,不可以被修改,如果不加const,那么权限被放大,所以需要加上const保证他的权限不变。

其实现在ra的地址已经不再是原变量a的地址了,是其中临时变量的地址。

1.4:使用场景

做参数:

void Swap(int& x, int& y)
{
	int tmp = x;
	x = y;
	y = tmp;
}
void Swap(double& x, double& y)
{
	double tmp = x;
	x = y;
	y = tmp;
}

做返回值 下面我们先看两个代码

int & Add(int a, int b){
	static int c = a + b;
	return c;
}
int main(){
	int & ret = Add(1, 2);
	Add(3, 4);
	cout << "Add(1, 2) is :" << ret << endl;
	system("pause");
	return 0;
}

Add(1, 2) is :3
//请按任意键继续. . .

int & Add(int a, int b){
	int c = a + b;
	return c;
}
int main(){
	int & ret = Add(1, 2);
	Add(3, 4);
	cout << "Add(1, 2) is :" << ret << endl;
	system("pause");
	return 0;
}

Add(1, 2) is :7
//请按任意键继续. . .

都一个结果是3,第二个结果是7,为什么会这样呢?

我们知道在函数返回值时实际是产生一个临时变量,若传值返回,那么实际是发生了拷贝,若引用返回,那么其实是给这个临时变量取了别名,返回了这个临时变量的别名。 对于静态变量c的作用域不变,但是生命周期变长,在Add函数返回时,出了作用域,返还对象并没有还给系统,所以此时ret一直是第一次调用Add函数时产生的临时变量的别名,所以ret的结果是3。

1.5:引用和指针的区别

在语法概念上,引用就是一个别名,没有开辟独立的空间存储,和其引用实体共用同一块实体。

int main(){
	int a = 10;
	int & ra = a;
	cout << "&a = " << &a << endl;
	cout << "&ra = " << &ra << endl;
	system("pause");
	return 0;
}

&a = 0137F8D8
&ra = 0137F8D8
请按任意键继续. . .

由代码结果可看是同一块地址。

在底层实现上实际是有空间的,因为引用是按照指针方式来实现的。

引用和指针的不同点:

  • 引用在定义时必须初始化,指针没有要求。
  • 引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以在任何时候指向任何一个同类型实体。
  • 没有NULL引用,但是有NULL指针。
  • sizeof中含义不同:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数。
  • 引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小。
  • 有多级指针,但没有多级引用。
  • 访问实体不同,指针需要显式解引用,引用编译器自己处理。
  • 引用比指针使用起来更加安全。

二:内联函数

2.1:概念

以inline修饰的函数叫做内联函数,用于解决C语言中宏函数难懂易错的缺陷。在编译时C++编译器会在调用内联函数的地方展开,没有函数压栈的开销,内联函数提升程序运行的效率。

如果在上述函数前增加inline关键字,将其改成内联函数,那么在编译期间编译器会用函数体(展开)替换函数的调用。

查看方式:

在Release模式下,查看编译器生成的汇编代码中是否有call Add

在Debug模式下,需要对编译器进行设置,否则也不会展开。

如VS2013版本设置:

右击项目名称——>属性:

2.2:特性

  • inline是一种以空间换取时间的做法,利用直接展开函数省去调用函数的开销。所以对于代码很长或者有循环/递归的函数不方便使用作为内联函数。
  • inline对于编译器而言只是一个建议,编译器会自动优化,如果定义的inline的函数体内有循环/递归等,编译器优化时会忽略掉内联。
  • inline不建议声明和定义分开,分开会导致链接错误,因为inline被展开,就没有函数地址了,如果分开了,链接的时候就找不到了。

2.3:面试题

宏的优点?

  • 增强代码的复用性
  • 提高性能

缺点:

  • 不方便调试,因为编译阶段进行了宏替换。
  • 导致代码可读性差,可维护性差,容易误用。
  • 没有类型安全检查。

C++哪些技术可以替代宏?

  • 常量定义,换用const。
  • 函数定义,换用内联函数。

三:auto关键字

3.1:auto简介

我们在学习C语言的时候就见过auto,当时认为使用auto修饰的变量是具有自动存储器的局部变量,但我们几乎没有使用。

在C++11中,标准委员会赋予了auto全新的含义:auto不再是一个存储类型标识符,而是作为一个新的类型指示符来指示编译器,auto声明的变量必须由编译器在编译时期推导而得。

int TestAuto()
{
	return 10;
}
int main()
{
	const 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;
}

这里我们看打印的结果是:

int const *
char
int
请按任意键继续. . .

可见auto关键字可以自动识别变量的类型。这里**typeid(b).name()**可返回变量类型的字符串。

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

3.2:auto使用细则

auto与指针和引用结合使用

用auto声明指针类型时,用auto和auto*没有任何区别,但是用auto声明引用类型时则必须加&。

int main()
{
	int x = 10, y = 20;
	auto a = &x;
	auto* b = &x;
	auto& c = x;
	cout << typeid(a).name() << endl;
	cout << typeid(b).name() << endl;
	cout << typeid(c).name() << endl;
	*a = 20;
	*b = 30;
	c = 40;
	system("pause");
	return 0;
}

int *
int *
int
请按任意键继续. . .

看这里a和b的类型都是int * ,所以用auto声明指针类型时,用auto和auto*没有任何区别。

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

注意看d下面编译器报错了,因为c和d的初始化表达式类型不同。

3.3:auto不能推导的场景

1:auto不可以作为函数形参

因为在形参处使用auto,编译器无法对a的实际类型进行推导。

2:auto不可以用来声明数组。

四:基于范围的for循环

4.1:范围for循环的语法

对于一个有范围的集合而言,在C++98中由程序员来说明循环的范围是多余的,有时候还容易犯错。因此在C++11中引入基于范围的for循环。for循环后的括号由冒号“:”分为两部分。第一部分是范围内用于迭代的变量,第二部分则是表示被迭代的范围。

int main(){
	int array[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9};
	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;
	// 范围for
	// 依次自动取array中的数据,赋值给e,自动判断结束
	for (auto& e : array){
		e /= 2;
	}
	for (auto x : array){
		cout << x << " ";
	}
	cout << endl;
	system("pause");
	return 0;
}

2 4 6 8 10 12 14 16 18
1 2 3 4 5 6 7 8 9
请按任意键继续. . .

对于语句auto x : array意思是依次自动取array中的数据,赋值给e,所以相当于e是array中的拷贝,既然是拷贝,也就不能对array中的数据产生影响。对要改变array中数据,需要利用引用,正如代码中auto& e : array。

4.2:范围for循环的使用条件

for循环迭代的范围必须是确定的。

下面给出一个错误代码示例:

void TestFor(int array[])
{
	for (auto& e : array)
		cout << e << endl;
}

因为我们知道,函数传参,形参相对于实参发生降维,形参array是一个整型指针,所以这里for循环的迭代范围是不确定的。

到此这篇关于C++类与对象深入之引用与内联函数与auto关键字及for循环详解的文章就介绍到这了,更多相关C++类与对象内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C++超详细讲解引用和指针

    目录 引用概念 定义步骤 引用必须初始化 引用初始化后不能更改 引用作为函数的参数可以替代指针变量 常引用 引用作为函数的返回值类型 引用的本质 指针的引用(了解) 指针和引用的区别 引用概念 引用的本质:给已有的变量名 取个别名 //给num取个别名为b int num =100; //&不是取b的地址 只是描述b是num的别名 编译器不会为b开辟新的空间 int &b = num;//num的别名 是b //操作b等价操作num 定义步骤 1.&修饰别名 2.给哪个变量取别名

  • C++超详细分析讲解内联函数

    目录 宏函数(带参数的宏)的缺点 inline修饰的函数就是内联函数 内联函数的特点 宏函数和内联函数的区别 宏函数(带参数的宏)的缺点 第一个问题:宏函数看起来像一个函数调用,但是会有隐藏一些难以发现的问题. 例如: #define FUN(x, y) (x * y) printf("%d", add(3, 3 + 2)) //3 * 3 + 2 = 11 以上情况可以通过加 “()” 解决: #define FUN(x, y) (x * y) printf("%d&quo

  • C++11关于auto关键字的使用示例

    一.概述 auto关键字在c++98中已经出现,在98中定义为具有自动存储器的局部变量, c++11中标准委员会重新定义了auto关键字,表示一个类型占位符,告诉编译器,auto声明变量的类型必须由编译器在编译时期推导 而得. 注意事项: 1.auto关键字类型推断发生在编译期,程序运行时不会造成效率降低 2.auto关键字定义时就需要初始化 3.auto仅仅是一个占位符,它并不是一个真正的类型, 因此sizeof(auto)是错误的 4.auto不能作为函数的参数 5.auto不能定义数组,如

  • C++深入探索内联函数inline与auto关键字的使用

    目录 1.内敛函数 1.1问题引入 1.2内联函数的概念 1.3内敛函数的特性 2.auto关键字 2.1 auto简介 2.2 auto的使用细则 2.3 auto不能推导的场景 2.4 auto与新式for循环使用 1.内敛函数 1.1问题引入 我们在使用C语言中我们都学过函数,我们知道函数在调用的过程中需要开辟栈帧.如果我们需要频繁的调用一个函数,假设我们调用10次Add()函数,那我们就需要建立10次栈帧.我们都知道在栈帧中要做很多事情,例如保存寄存器,压参数,压返回值等等,这个过程是很

  • C++ 引用与内联函数详情

    目录 引用初阶 什么是引用 为何要有引用 引用指向同一块空间 引用的特性 定义时必须初识化 一个变量可以多次引用 引用一旦引用了一个实例,不能在再引用其他的实例 引用进阶 常引用 权限 临时变量具有常属性 引用的场景 做参数 返回值 引用做返回值 引用不会开辟空间 引用和指针比较 内联函数 为何存在 内联函数 展开短小的函数 内联函数的特性 较大的函数编译器不会发生内联 声明定义一起 引用初阶 引用是C++的特性的之一,不过C++没有没有给引用特意出一个关键字,使用了操作符的重载.引用在C++中

  • C++深入分析内联函数的使用

    目录 一.常量与宏回顾 二.内联函数 三.内联函数使用注意事项 四.小结 一.常量与宏回顾 C++中的const常量可以替代宏常数定义,如︰ 但是C++中是否有解决方替代宏代码片段呢?这里就要引入内联函数. 二.内联函数 C++中推荐使用内联函数替代宏代码片段 C++中使用 inline 关键字声明内联函数 内联函数声明时inline关键字必须和函数定义结合在一起,否则编译器会直接忽略内联请求 C++编译器可以将一个函数进行内联编译 被C++编译器内联编译的函数叫做内联函数 C++编译器直接将函

  • C++深入讲解引用的特点及与指针的区别

    目录 一.引入 二.C++中较为麻烦的运算符 三.引用的定义 四.引用的特点 五.对比指针与引用 六.引用与指针的区别 1.语法层面的区别 2.汇编层面的区别 七.引用的其他使用 常引用 数组引用 指针引用 一.引入 在生活中,我们可能也会给一些同学起外号,以“张磊”同学为例,我们可以叫他“张三石”,当我们叫到这个外号的时候就会自然而然的想到“张磊”同学,”张三石”就是张磊的别名,而引用也可以这样简单理解:在语法层面上,引用就是取别名. 二.C++中较为麻烦的运算符 C++中的 * 和 & 有多

  • C++实例讲解引用的使用

    目录 1.什么是引用 2.引用的用法 2.1 普通引用 2.2 const 引用 2.3 作用在函数参数 2.4 作用在函数返回值 3.引用的本质 1.什么是引用 引用可以看作是一个已经定义的变量的别名,其作为变量别名而存在. 2.引用的用法 2.1 普通引用 当引用作为普通引用时,其使用语法如下所示: Type& new_name = old_name; e.g. int a = 0; int& b = a; b = 5; // 此时,a = 5 注意 在定义时必须使用相同类型的变量进行

  • C++类与对象深入之引用与内联函数与auto关键字及for循环详解

    目录 一:引用 1.1:概念 1.2:引用特性 1.3:常引用 1.4:使用场景 1.5:引用和指针的区别 二:内联函数 2.1:概念 2.2:特性 2.3:面试题 三:auto关键字 3.1:auto简介 3.2:auto使用细则 3.3:auto不能推导的场景 四:基于范围的for循环 4.1:范围for循环的语法 4.2:范围for循环的使用条件 一:引用 1.1:概念 引用不是定义一个新的变量,而是给已经存在的变量取一个别名.注意:编译器不会给引用变量开辟内存空间,他和他的引用变量共用同

  • C++入门(命名空间,缺省参数,函数重载,引用,内联函数,auto,范围for)

    一.C++关键字 C++总共有63个关键字,在入门阶段我们只是大致了解一下就可,在后续博客中会逐渐讲解 二.命名空间 相信学过C++的同学,一定都写过下面这个简单的程序 #include<iostream> using namespace std; int main() { cout<<"hello world"<<endl; return 0; } 我们先来看第二行代码,using namespace std , 这行代码是什么意思呢 ? 这里我们

  • C++ 类中有虚函数(虚函数表)时 内存分布详解

    虚函数表 对C++ 了解的人都应该知道虚函数(Virtual Function)是通过一张虚函数表(Virtual Table)来实现的.简称为V-Table.在这个表中,主是要一个类的虚函数的地址表,这张表解决了继承.覆盖的问题,保证其容真实反应实际的函数.这样,在有虚函数的类的实例中这个表被分配在了这个实例的内存中,所以,当我们用父类的指针来操作一个子类的时候,这张虚函数表就显得由为重要了,它就像一个地图一样,指明了实际所应该调用的函数. 这里我们着重看一下这张虚函数表.C++的编译器应该是

  • C++示例分析内联函数与引用变量及函数重载的使用

    目录 1.内联函数 1.1为什么使用内联函数 1.2语法 2.引用变量 2.1为什么要使用引用变量 2.2语法 2.3对于C语言的改进 3. 函数重载 3.1默认参数 3.2函数重载 1.内联函数 1.1为什么使用内联函数 减少上下文切换,加快程序运行速度. 是对C语言中的宏函数的改进. 1.2语法 #include<iostream> using namespace std; inline double square(double x){ return x*x; } int main(){

  • C++类和对象实例解析(二)

    C++既是面向对象也是面向过程的语言,在这里就有一个重要的概念--类.         何谓类?类是对对象的一种抽象,举例来讲:每一个实实在在存在的人就是一个对象,人有很多共同的特征(一个头,两条腿,能走,能跑),这具有共同特征的人就成为一个类.类是一个抽象的名词,每一个人(即对象)是这个类的实例.         对象间具有的共同特征是对象的属性和行为.录像机是一个对象,它的属性是生产厂家.牌子.重量.颜色等等,它的行为就是它的功能,如录像.放像.快进.倒退等操作. C++程序中,需要先定义一

  • 讲解C#面相对象编程中的类与对象的特性与概念

    类 "类"是一种构造,通过使用该构造,您可以将其他类型的变量.方法和事件组合在一起,从而创建自己的自定义类型.类就像一个蓝图,它定义类型的数据和行为.如果类没有声明为静态类,客户端代码就可以创建赋给变量的"对象"或"实例",从而使用该类.在对变量的所有引用都超出范围之前,该变量始终保持在内存中.所有引用都超出范围时,CLR 将标记该变量以供垃圾回收.如果类声明为静态类,则内存中只存在一个副本,并且客户端代码只能通过该类自身而不是"实例变

  • 带你用Java全面剖析类和对象

    目录 一.面向过程?面向对象? 二.类和类的实例化 2.1普通成员变量和普通成员方法 2.2 静态成员变量和静态成员方法 三.封装 3.1 private 3.2 getter 和 setter 四.构造方法 4.1 基本语法 4.2 this 关键字 五.代码块 5.1 普通代码块 5.2 构造代码块 5.3 静态代码块 5.4 注意事项 六.快捷方法 6.1 toString方法 6.2 setter / getter 方法 6.3 构造方法(快捷) 总结 一.面向过程?面向对象? C 语言

  • C++初识类和对象

    目录 一.初步认识面向过程和面向对象 二.类的引入 三.类的定义 1.定义和声明全部放在类体中,需要注意的是: 2.声明与定义分离 四.类的访问限定符及封装 1.访问限定符 2.封装 五.类的作用域 六.类的实例化 七.类对象模型 1.计算类对象的大小 2.类对象的存储方式 八.this指针 1.this指针的引出 2.this指针的特性 总结 一.初步认识面向过程和面向对象 面向过程,关注的是怎么去做,比如在外卖系统中,强调点餐,做餐,送餐等一系列动作的方法,反映到语言中是函数方法的实现:而面

  • C++类和对象实战之Date类的实现方法

    目录 零.前言 一.Date类相关接口 二.具体接口函数实现 1.获取月份天数 2.Date打印 3.Date构造函数 4.Date析构函数 5.Date拷贝构造函数 6.Date赋值重载函数 7.Date+=天数 8.Date+天数 9.Date-=天数 10.Date-天数 11.++Date 12.Date++ 13.–Date 14.Date– 15.日期比较 16.Date相减 17.日期输入\日期输出 总结 零.前言 在学了C++类和对象基本知识以及六个默认成员函数后,我们可以上手实

随机推荐