免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 3517 | 回复: 11
打印 上一主题 下一主题

[C++] 讨论Visual Studio 2010引入的C++0x特性之右值引用  关闭 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2009-11-17 17:20 |只看该作者 |倒序浏览
作为最重要的一项语言特性,右值引用(rvalue references)被引入到 C++0x中。我们可以通过操作符“&&”来声明一个右值引用,原先在C++中使用“&”操作符声明的引用现在被称为左值引用。
   1.  int a;
   2. int& a_lvref = a;  // 左值引用
   3.  
   4. int b;
   5. int&& b_rvref = b;  // 右值应用

左值引用和右值引用的表现行为基本一致,它们唯一的差别就是右值引用可以绑定到一个临时对象(右值)上,而左值引用不可以。例如:
   1.  int& a_lvref = int();      // error C2440: 'initializing' : cannot convert from 'int' to 'int &'   
   2. int&& b_rvref = int();  // OK!

在第一行代码中,我们将一个临时对象int()绑定到一个左值引用,将产生一个编译错误。而在第二行中,我们将临时对象绑定到右值引用,就可以顺利通过编译。
右值是无名的数据,例如函数的返回值一般说来就是右值。当对右值进行操作的时候,右值本身往往没有必要保留,因此在某些情况下可以直接“移动”之。通过右值引用,程序可以明确的区分出传入的参数是否为右值,从而避免了不必要的拷贝,程序的效率也就得到了提高。我们考虑一个简单的数据交换的小程序,从中来体会右值引用所带来的效率提升。我们可以写一个函数swap来实现两个变量值的交换:
   1.  template <class T> swap(T& a, T& b)
   2. {
   3.     T tmp(a);   // tmp对象创建后,我们就拥有了a的两份拷贝
   4.     a = b;      // 现在我们拥有b的两份拷贝
   5.     b = tmp;    // 现在我们拥有a的两份拷贝
   6. }

在这段代码中,虽然我们只是为了进行简单的数据交换,但是却执行了多次对象拷贝。这些对象的拷贝操作,特别是当这些对象比较大的时候,无疑会影响程序的效率。
那么,如果使用右值引用如何实现呢?
   1.  // RValueRef.cpp : Defines the entry point for the console application.
   2. //
   3.  
   4. #include "stdafx.h"
   5.  
   6. template <class T>
   7. T&& move(T&& a)
   8. {
   9.     return a;
  10. }
  11.  
  12. template <class T> void swap(T& a, T& b)
  13. {
  14.     T tmp(move(a)); // 对象a被移动到对象tmp,a被清空
  15.     a = move(b);    // 对象b被移动到对象a,b被清空
  16.     b = move(tmp);  // 对象tmp被移动到对象b
  17. }
  18.  
  19. int _tmain(int argc, _TCHAR* argv[])
  20. {
  21.     int a = 1;
  22.     int b = 2;
  23.     swap(a, b);
  24.  
  25.    return 0;
  26. }

在这段重新实现的代码中,我们使用了一个move()函数来代替对象的赋值操作符“=”,move()只是简单地接受一个右值引用或者左值引用作为参数,然后直接返回相应对象的右值引用。这一过程不会产生拷贝(Copy)操作,而只会将源对象移动(Move)到目标对象。
正是拷贝(Copy)和移动(Move)的差别,使得右值引用成为C++0x中最激动人心的新特性之一。从实践角度讲,它能够完美是解决C++中长久以来为人所诟病的临时对象的效率问题。从语言本身讲,它健全了C++中的引用类型在左值右值方面的缺陷。从库设计者的角度讲,它给库设计者又带来了一把利器。而对于广大的库使用者而言,不动一兵一卒便能够获得“免费的”效率提升.
在Visual Studio 2010中,因为有了对这些C++0x新特性的支持,重新点燃了程序员们对C++的热情。C++重振雄风,指日可待!


没用过有点不是很理解.

论坛徽章:
1
2015年迎新春徽章
日期:2015-03-04 09:49:45
2 [报告]
发表于 2009-11-17 17:24 |只看该作者

论坛徽章:
0
3 [报告]
发表于 2009-11-17 18:47 |只看该作者
重载,多态,编译时的代价多大还搞什么右值引用

论坛徽章:
2
青铜圣斗士
日期:2015-11-26 06:15:59数据库技术版块每日发帖之星
日期:2016-07-24 06:20:00
4 [报告]
发表于 2009-11-17 18:47 |只看该作者
兄弟,修改一行代码并去掉一句话,这文章不错。

原帖由 dhb118 于 2009-11-17 17:20 发表

int _tmain(int argc, _TCHAR* argv[])

在Visual Studio 2010中,因为有了对这些C++0x新特性的支持,重新点燃了程序员们对C++的热情。C++重振雄风,指日可待!


否则,我的反应也和楼上差不多……


修改一下,是和楼上上差不多。
被插队了……

[ 本帖最后由 OwnWaterloo 于 2009-11-17 18:49 编辑 ]

论坛徽章:
0
5 [报告]
发表于 2009-11-17 18:53 |只看该作者
其实俺更希望MS想办法把console和windows两个链接类型合并

另外问问楼上的大牛们,右值引用能够实现闭包和lambda吗

论坛徽章:
0
6 [报告]
发表于 2009-11-17 18:55 |只看该作者

回复 #4 OwnWaterloo 的帖子

还有就是问下兄弟,看了google的GO语言了吗,有何感想

论坛徽章:
2
青铜圣斗士
日期:2015-11-26 06:15:59数据库技术版块每日发帖之星
日期:2016-07-24 06:20:00
7 [报告]
发表于 2009-11-17 18:59 |只看该作者

回复 #5 reiase 的帖子

右值引用和闭包和lambda有什么关系吗?
闭包和lambda又是什么关系? lambda表达式是产生闭包的一种方式?

木有什么感想……
又发明一个语言作啥呢……
代替python? 不会是想代替C吧……

论坛徽章:
0
8 [报告]
发表于 2009-11-17 19:01 |只看该作者
go就是C++和Python的杂交

论坛徽章:
2
青铜圣斗士
日期:2015-11-26 06:15:59数据库技术版块每日发帖之星
日期:2016-07-24 06:20:00
9 [报告]
发表于 2009-11-17 19:02 |只看该作者

回复 #6 reiase 的帖子

我对vim作者那个什么语言…… 有点感想……

"既可以写kernel 又可以写脚本",不知道是不是作者自己吹的,应该不是……
很扯谈……


IF ...
} // 前面没有配套的 "{ "很不习惯…… 当然,这是小问题。
大问题是IF很难敲……

论坛徽章:
2
青铜圣斗士
日期:2015-11-26 06:15:59数据库技术版块每日发帖之星
日期:2016-07-24 06:20:00
10 [报告]
发表于 2009-11-17 19:08 |只看该作者

回复 #6 reiase 的帖子

更正一下7楼的说法。

没有什么感想是因为基本没去看,只看了看cnblogs上的一些杂文……
没去看的原因就是7楼说的, 不知道出个语言做什么……
还有一个原因是……  我要做墙内的好公民, 哈哈哈。
据说新的golang.org没被墙?

vim那个是因为vim名气太大了…… 就去看了看……
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP