ImageVerifierCode 换一换
格式:PPT , 页数:38 ,大小:215KB ,
资源ID:4839985      下载积分:10 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.docduoduo.com/d-4839985.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录   微博登录 

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(孙鑫VC++教程之第04节补充-引用.ppt)为本站会员(fmgc7290)主动上传,道客多多仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知道客多多(发送邮件至docduoduo@163.com或直接QQ联系客服),我们立即给予删除!

孙鑫VC++教程之第04节补充-引用.ppt

1、引用的扩展,引用和指针的区别引用的补充说明const &(const引用)数组的引用,引用和指针的区别,指针和引用主要有两个区别: 1.引用必须总是指向一个对象。 int *pi = NULL; 表示用NULL初始化pi,pi不指向任何对象。 const int ,引用和指针的区别,2.引用之间的赋值是另外一个不同,如果用一个引用给另一个引用赋值,那么改变的是被引用对象而不是引用本身。 int ival = 1024,ival2 = 2048; int *pi = 表示什么呢? pi指向的对象ival并没有改变。实际上pi被赋值为指向pi2所指向的对象,即ival2,现在pi和pi2指向同一个

2、对象。,引用和指针的区别,int ival = 1024,ival2 = 2048; int 表示什么?改变的是ival,而引用本身没有变化,赋值之后,两个引用仍然指向原来的对象。,引用和指针的区别,实际上c+很少独立的使用引用。引用主要用在函数的形参。bool GetNextValue(int ,引用的补充说明,#include #include using namespace std; void main(int argc,char* argv) int a=10; int b=20; int ,引用的补充说明,由于引用本身就是目标的一个别名,引用本身的地址是一个没有意义的值,所以在c+中

3、是无法取得引用的内存地址的。取引用的地址就是取目标的地址,c+本身就根本不提供获取引用内存地址的方法。 引用一旦初始化,就不在能够被指向其它的目标,虽然编译不会出错,但操作是不起作用的,实际上还是指向最先指向的目标。上面代码中的rn=b实际在计算机看来就是a=b,所以修改的还是a的值。,引用的补充说明,#include #include using namespace std; void main(int argc,char* argv) int a=10; void ,引用的补充说明,上面的两错误要记住引用的特性: void修饰是不能够声明引用的 引用是不能够声明数组的,即不能够声明引用数组

4、。为什么? 引用的一个重要的特性是,声明引用的同时要进行初始化,而数组元素的初始化方式对引用来说是没有意义的。,引用的补充说明(1),(1)全局返回变量值 float c; float test(float,float); void main(int argc,char* argv) float pn=test(3.0f,1.2f); coutpn; cin.get(); float test(float a,float b) c=a*b; return c; ,引用的补充说明,在上面的代码中我们可能以为函数返回的就是c变量。这么想可能就错了,普通情况下我们在函数内进行普通值返回的时候在内存栈

5、空间内其实是自动产生了一个临时变量temp,它是返回值的一个副本一个copy,函数在return的时候其实是return的这个临时产生的副本。,引用的补充说明,数据在内存中的情况如下图:,(2)下面我们再来看一种情况,就是把返回值赋给引用: #include #include using namespace std; float c; float test(float,float); void main(int argc,char* argv) float ,引用的补充说明,float 这句在bc中能够编译通过,因为bc扩展设置为临时变量设置引用,那么临时变量的生命周期将和引用的生命周期一致。

6、 但在vc中却不能通过编译,因为一但test()执行过后临时变量消失在栈空间内,这时候pn将成为一个没有明确目标的引用,会导致内存出错,引用的补充说明,它在内存中的情况见下图:,我们在图中看到,由于函数仍然是普通方法返回,所以仍然会有一个副本临时变量产生,只不过,这一次只是返回一个目标地址,在main中目标地址被赋予了引用pn。,下面我们再看一种情况,这是返回引用给变量的情况: #include #include using namespace std; float c; float ,引用的补充说明,这种返回引用给变量的情况下,在内存中,test()所在的栈空间内并没有产生临时变量,而是直接

7、将全局变量c的值给了变量pn,这种方式是我们最为推荐的操作方式,因为不产生临时变量直接赋值的方式可以节省内存空间提高效率,程序的可读性也是比较好的。,引用的补充说明,它在内存中的情况见下图:,最后的一种情况是函数返回引用,并且发值赋给一个引用的情况 #include #include using namespace std; float c; float ,引用的补充说明,这种情况同样也不产生临时变量,可读和性能都很好,但有一点容易弄错,就是当c是局部变量或者是在堆内存中临时开辟后来又被free掉了以后的区域,这种情况和返回的指针是局部指针的后果一样严重,会导致引用指向了一个不明确的地址,引用

8、的补充说明,在内存中情况见下图:,由于这种情况存在作用域的问题,故我们推荐采用第三种方式处理。,引用的补充说明,接下来我们说几个利用引用作为左值参与计算的例子,这一点一非常重要,对于理解返回引用的函数是非常有帮助的。,float c; float ,引用的补充说明,通常来说函数是不能作为左值,因为引用可以做为左值,所以返回引用的函数自然也就可以作为左值来计算了。 在上面的代码中:float 把函数作左值进行计算,这里由于test是返回引用的函数,其实返回值返回的地址就是c的地址,自然c的值就被修改成了12.1。,const &(const引用),const引用可以用不同类型的对象初始化(只要能

9、从一种类型转换到另一种类型即可),也可以是不可寻址的值,如文字常量,表达式值等。例如: double dval = 3.14; /仅仅对于cosnt才是合法的 const int ,const &(const引用),同样的初始化对于非const引用是不合法的,将导致编译错误。原因有些微妙,需要适当的做些解释。 引用在内部存放的是一个对象的地址,是该对象的别名。对于不可寻址的值,如文字常量,以及不同类型的对象,编译器为了实现引用,必须生成一个临时对象,引用实际上指向该对象,但用户不能访问它。 double dval =1024; const int ,const &(const引用),如果我们给

10、ri赋值,不会改变dval的值,而是改变temp。对用户来说,修改动作没有生效,是没有意义的。因此使用const来进行约束,去除这种无意义的行为,因为const表示只读。不允许非const的引用指向需要临时对象的对象或者值。这比“允许定义这样的引用,但实际不会生效”的方案要好。,const &(const引用),下面我们通过一个例子来说明。 希望将一个引用初始化为一个const对象的地址。 const int ival = 1024;首先,我们知道非const引用定义是不合法的,将导致编译错误。 int /错误,要求一个const引用,const &(const引用),const int *,

11、const &(const引用),通过下面的对比,我们就可以更加明白为什么要用上面的方式表示了。 /声明一个指针const int ival = 1024;const int *pival = ,数组的引用,在C+中可以定义数组的引用,用以解决C中无法解决的“数组降阶”问题,我们先来看看什么是“数组降阶”,先看如下代码: void Test( char array20 ) cout sizeof(array) endl;/输出? char array20 = 0 ; cout sizeof(array) endl; /输出? Test( array );,数组的引用,我们看到,对于同样的数组a

12、rray,一个输出4,另一个输出20 。 这是因为void Test( char array20 ) 中的array被降阶处理了, void Test( char array20 ) 等同于 void Test( char array ) 也等同于 void Test( char* const array ) 如果你原意,它甚至等同于 void Test( char array999 ),数组的引用,也就是说 void Test( char array20 ) cout sizeof(array) endl; 被降成 void Test( char* const array ) cout si

13、zeof(array) endl; / 既然是char*,当然输出4 ,数组的引用,这样以来,我们在函数声明中的数组大小限制是无效的,声明 void Test( char array20 ) 并不能保证一定会接收到一个大小20的数组,即任何 char 都会被降价为 char* ,这样就增加了程序出错的可能性。要解决这样一个问题,我们可以用C+的数组引用作为参数,看以下代码:,数组的引用,void Test( char (,数组的引用,这样 Test 函数就只能接收大小为 20 的 char ,看如下代码: char array110 = 0 ; char array220 = 0 ; Test

14、(array1);/Error:实参不是大小为 10 的 char Test(array2);/OK,数组的引用,在 C+ 中,单纯的用数组的引用可以直接传递数组名,因为它将数组的大小已在形参里提供了信息。但是这样一来我们只能固定数组的大小来用这个函数了。用模板加数组的引用可以解决这个问题,看如下代码: template void Test(int (/OK,数组的引用,使用模板后确实可以使同一函数能够处理大小不同的数组了,扩大了函数的适用范围。但是这样定义的函数仍然存在着下述缺点: 1. 模板最终是要实例化的,所以调用多少个不同长度的数组,就要产生这个函数的多少份实例代码。而传统方式的函数只有一份实例,与函数的调用次数无关。 2. 不能应用于在编译期间数组的大小尚未确定的情况,这也使这个模板函数的适用范围受到限制。 3. 这样写的函数显然不能用指针变量作为函数的参数,因此不能用这个函数处理动态分配的内存区域。,

本站链接:文库   一言   我酷   合作


客服QQ:2549714901微博号:道客多多官方知乎号:道客多多

经营许可证编号: 粤ICP备2021046453号世界地图

道客多多©版权所有2020-2025营业执照举报