成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Visual Studio 2010中C++的四大變化

開發 后端
微軟即將在2010年4月12日發布Visual Studio 2010的正式版,對于C++語言做了修改,使之更加符合C++標準,文章將對C++語言的修改來做一下分析。

在微軟即將發布的Visual Studio 2010正式版中,其對C++語言做了一些修改,之前51cto也報道過Visual Studio 2010中關于C++項目的升級問題,文章則針對C++語言上的一些變化。

Lambda表達式

很多編程編程語言都支持匿名函數(anonymous function)。所謂匿名函數,就是這個函數只有函數體,而沒有函數名。Lambda表達式就是實現匿名函數的一種編程技巧,它為編寫匿名函數提供了簡明的函數式的句法。同樣是Visual Studio中的開發語言,Visual Basic和Visual C#早就實現了對Lambda表達式的支持,終于Visual C++這次也不甘落后,在Visual Studio 2010中添加了對Lambda表達式的支持。

Lambda表達式使得函數可以在使用的地方定義,并且可以在Lambda函數中使用Lambda函數之外的數據。這就為針對集合操作帶來了很大的便利。在作用上,Lambda表達式類似于函數指針和函數對象,Lambda表達式很好地兼顧了函數指針和函數對象的優點,卻沒有它們的缺點。相對于函數指針或是函數對象復雜的語法形式,Lambda表達式使用非常簡單的語法就可以實現同樣的功能,降低了Lambda表達式的學習難度,避免了使用復雜的函數對象或是函數指針所帶來的錯誤。我們可以看一個實際的例子:

  1. #include "stdafx.h"   
  2. #include <algorithm>   
  3. #include <iostream>   
  4. #include <ostream>   
  5. #include <vector>   
  6.  
  7. using namespace std;   
  8.  
  9. int _tmain(int argc, _TCHAR* argv[])   
  10. {   
  11. vector<int> v;   
  12. for (int i = 0; i < 10; ++i) {   
  13. v.push_back(i);   
  14. }   
  15.  for_each(v.begin(), v.end(), [] (int n) {   
  16. cout << n;   
  17. if (n % 2 == 0) {   
  18. cout << " even ";   
  19. } else {   
  20. cout << " odd ";   
  21. }   
  22. });   
  23. cout << endl;   
  24.  
  25. return 0;   
  26. }  
  27. #include "stdafx.h"  
  28. #include <algorithm> 
  29. #include <iostream> 
  30. #include <ostream> 
  31. #include <vector> 
  32.  
  33. using namespace std;  
  34.  
  35. int _tmain(int argc, _TCHAR* argv[])  
  36. {  
  37. vector<int> v;  
  38. for (int i = 0; i < 10; ++i) {  
  39. v.push_back(i);  
  40. }  
  41.  for_each(v.begin(), v.end(), [] (int n) {  
  42. cout << n;  
  43. if (n % 2 == 0) {  
  44. cout << " even ";  
  45. } else {  
  46. cout << " odd ";  
  47. }  
  48. });  
  49. cout << endl;  
  50.  
  51. return 0;  

這段代碼循環遍歷輸出vector中的每一個數,并判斷這個數是奇數還是偶數。我們可以隨時修改Lambda表達式而改變這個匿名函數的實現,修改對集合的操作。在這段代碼中,C++使用一對中括號“[]”來表示Lambda表達式的開始,其后的”(int n)”表示Lambda表達式的參數。這些參數將在Lambda表達式中使用到。為了體會Lambda表達式的簡潔,我們來看看同樣的功能,如何使用函數對象實現:

  1. #include "stdafx.h"   
  2. #include <algorithm>   
  3. #include <iostream>   
  4. #include <ostream>   
  5. #include <vector>   
  6. using namespace std;   
  7.  
  8. struct LambdaFunctor {   
  9. void operator()(int n) const {   
  10. cout << n << " ";   
  11. if (n % 2 == 0) {   
  12. cout << " even ";   
  13. } else {   
  14. cout << " odd ";   
  15. }   
  16.  
  17. }   
  18. };   
  19.  
  20. int _tmain(int argc, _TCHAR* argv[])   
  21. {   
  22. vector<int> v;   
  23.  
  24. for (int i = 0; i < 10; ++i) {   
  25. v.push_back(i);   
  26. }   
  27.  
  28. for_each(v.begin(), v.end(), LambdaFunctor());   
  29. cout << endl;   
  30.  
  31. return 0;   
  32. }  
  33. #include "stdafx.h"  
  34. #include <algorithm> 
  35. #include <iostream> 
  36. #include <ostream> 
  37. #include <vector> 
  38. using namespace std;  
  39.  
  40. struct LambdaFunctor {  
  41. void operator()(int n) const {  
  42. cout << n << " ";  
  43. if (n % 2 == 0) {  
  44. cout << " even ";  
  45. } else {  
  46. cout << " odd ";  
  47. }  
  48.  
  49. }  
  50. };  
  51.  
  52. int _tmain(int argc, _TCHAR* argv[])  
  53. {  
  54. vector<int> v;  
  55.  
  56. for (int i = 0; i < 10; ++i) {  
  57. v.push_back(i);  
  58. }  
  59.  
  60. for_each(v.begin(), v.end(), LambdaFunctor());  
  61. cout << endl;  
  62.  
  63. return 0;  
  64. }  


通過比較我們就可以發現,Lambda表達式的語法更加簡潔,使用起來更加簡單高效。 #p#

靜態斷言static_assert

在之前的C++標準C++03中,我們可以使用兩種斷言:

◆使用預處理中的條件編譯和#error指令,可以在預處理階段檢查一些編譯條件。
◆可以使用宏assert來進行運行時檢查,以確保程序邏輯的正確性。

但使用#error方法是非常煩瑣的,并且不能夠對模板參數進行檢查,因為模板實例化是在編譯時進行,而#error方法是在預處理階段進行的。而assert宏是在運行時進行檢查。不難發現,我們缺少了一樣東西,那就是可用于在編譯時檢查的工具。于是,靜態斷言應運而生。

在新的C++標準C++0x中,加入了對靜態斷言的支持,引入了新的關鍵字static_assert來表示靜態斷言。使用靜態斷言,我們可以在程序的編譯時期檢測一些條件是否成立,這個特性在調試模板函數的模板參數時特別有用。在編譯的時候,模板函數實例化,這時我們就可以使用靜態斷言去測試模板函數的參數是否按照我們的設計擁有合適的值。例如下面這段代碼:

  1. template <int N> struct Kitten {   
  2. static_assert(N < 2, "Kitten<N> requires N < 2.");   
  3. };   
  4.  
  5. int main() {   
  6. Kitten<1> peppermint;   
  7. Kitten<3> jazz;   
  8.  
  9.  return 0;   
  10. }  
  11. template <int N> struct Kitten {  
  12. static_assert(N < 2, "Kitten<N> requires N < 2.");  
  13. };  
  14.  
  15. int main() {  
  16. Kitten<1> peppermint;  
  17. Kitten<3> jazz;  
  18.  
  19.  return 0;  


當我們在主函數中使用“1”去實例化Kitten這個結構體時,在編譯的時候,靜態斷言static_assert會測試參數N的值,當N的值小于2時就會產生一個斷言錯誤,并將相應的調試幫助信息輸出到“Error List”窗口中,這樣程序員就可以對問題快速定位,解決問題就更加方便了。

另外,靜態斷言還帶來很多其他的優勢。例如靜態斷言在編譯時進行處理,不會產生任何運行時刻空間和時間上的開銷,這就使得它比assert宏具有更好的效率。另外比較重要的一個特性是如果斷言失敗,它會產生有意義且充分的診斷信息,幫助程序員快速解決問題。

auto關鍵字

在C++中,auto關鍵字的意義發生了改變。從Visual Studio 2010開始,auto關鍵字將用于指引編譯器根據變量的初始值來決定變量的數據類型。換句話說,我們可以把auto當成一種新的數據類型,它可以“從初始化器(initialize)中推導出所代表的變量的真正類型”。這種對auto關鍵字的使用方式可以大大消除當前替代方式所導致的冗長和易出錯的代碼。我們看一個實際的例子:

  1. #include <iostream>   
  2. #include <map>   
  3. #include <ostream>   
  4. #include <regex>   
  5. #include <string>   
  6. using namespace std;   
  7. using namespace std::tr1;   
  8.  
  9. int main() {   
  10. map<string, string> m;   
  11.  
  12. const regex r("(\\w+) (\\w+)");   
  13.  
  14. for (string s; getline(cin, s); ) {   
  15. smatch results;   
  16. if (regex_match(s, results, r)) {   
  17. m[results[1]] = results[2];   
  18. }   
  19. }   
  20. for (auto i = m.begin(); i != m.end(); ++i) {   
  21. cout << i->second << " are " << i->first << endl;   
  22. }   
  23.  
  24. return 0;   
  25. }  
  26. #include <iostream> 
  27. #include <map> 
  28. #include <ostream> 
  29. #include <regex> 
  30. #include <string> 
  31. using namespace std;  
  32. using namespace std::tr1;  
  33.  
  34. int main() {  
  35. map<string, string> m;  
  36.  
  37. const regex r("(\\w+) (\\w+)");  
  38.  
  39. for (string s; getline(cin, s); ) {  
  40. smatch results;  
  41. if (regex_match(s, results, r)) {  
  42. m[results[1]] = results[2];  
  43. }  
  44. }  
  45. for (auto i = m.begin(); i != m.end(); ++i) {  
  46. cout << i->second << " are " << i->first << endl;  
  47. }  
  48.  
  49. return 0;  

 
在這段代碼中,我們使用auto關鍵字來代替了真正的數據類型map<string, string>::iterator,這使得整個代碼自然而簡潔。另外,跟其他數據類型一樣,我們也可以對auto關鍵字進行修飾,例如添加const,指針(*),左值引用(&),右值引用(&&)等等,編譯器會根據auto類型所代表的真正的數據來決定這些修飾的具體含義。為了兼容一些舊有的C++代碼,我們可以使用/Zc:auto這個編譯器選項,來告訴編譯器是采用auto關鍵字的原有定義還是在新標準C++中的定義。 #p#

右值引用

作為最重要的一項語言特性,右值引用(rvalue references)被引入到 C++中。我們可以通過操作符“&&”來聲明一個右值引用,原先在C++中使用“&”操作符聲明的引用現在被稱為左值引用。

  1. int a;  
  2. int& aa_lvref = a;// 左值引用  
  3. int b;  
  4. int&& bb_rvref = b;// 右值應用 
  5.  

左值引用和右值引用的表現行為基本一致,它們唯一的差別就是右值引用可以綁定到一個臨時對象(右值)上,而左值引用不可以。例如:

  1. int& a_lvref = int();// error C2440: 'initializing' : cannot convert from 'int' to 'int &'  
  2. int&& b_rvref = int();// OK! 
  3.  

在***行代碼中,我們將一個臨時對象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的兩份拷貝   

在這段代碼中,雖然我們只是為了進行簡單的數據交換,但是卻執行了多次對象拷貝。這些對象的拷貝操作,特別是當這些對象比較大的時候,無疑會影響程序的效率。那么,如果使用右值引用如何實現呢?

  1. #include "stdafx.h"   
  2.  
  3. template <class T> 
  4. T&& move(T&& a)   
  5. {   
  6. return a;   
  7. }   
  8.  
  9. template <class T> void swap(T& a, T& b)   
  10. {   
  11. T tmp(move(a)); // 對象a被移動到對象tmp,a被清空   
  12. a = move(b);// 對象b被移動到對象a,b被清空   
  13. b = move(tmp);// 對象tmp被移動到對象b   
  14. }   
  15.  
  16. int _tmain(int argc, _TCHAR* argv[])   
  17. {   
  18. int a = 1;   
  19. int b = 2;   
  20. swap(a, b);   
  21.  
  22.  return 0;   
  23. }  
  24. #include "stdafx.h"  
  25.  
  26. template <class T>   
  27. T&& move(T&& a)  
  28. {  
  29. return a;  
  30. }  
  31.  
  32. template <class T> void swap(T& a, T& b)  
  33. {  
  34. T tmp(move(a)); // 對象a被移動到對象tmp,a被清空  
  35. a = move(b);// 對象b被移動到對象a,b被清空  
  36. b = move(tmp);// 對象tmp被移動到對象b  
  37. }  
  38.  
  39. int _tmain(int argc, _TCHAR* argv[])  
  40. {  
  41. int a = 1;  
  42. int b = 2;  
  43. swap(a, b);  
  44.  
  45.  return 0;  

 
在這段重新實現的代碼中,我們使用了一個move()函數來代替對象的賦值操作符“=”,move()只是簡單地接受一個右值引用或者左值引用作為參數,然后直接返回相應對象的右值引用。這一過程不會產生拷貝(Copy)操作,而只會將源對象移動(Move)到目標對象。

正是拷貝(Copy)和移動(Move)的差別,使得右值引用成為C++0x中最激動人心的新特性之一。從實踐角度講,它能夠***是解決C++中長久以來為人所詬病的臨時對象的效率問題。從語言本身講,它健全了C++中的引用類型在左值右值方面的缺陷。從庫設計者的角度講,它給庫設計者又帶來了一把利器。而對于廣大的庫使用者而言,不動一兵一卒便能夠獲得“免費的”效率提升,在Visual Studio 2010中,因為有了對這些C++新特性的支持,重新點燃了程序員們對C++的熱情。

【編輯推薦】

  1. C++指針漂移問題解決方案
  2. C++標準輸入輸出應用技巧解析
  3. C++托管程序實現安全管理
  4. C++內存對齊詳細使用指南
  5. C++二維數組初始化相關應用技巧分享
責任編輯:王曉東 來源: 博客園
相關推薦

2009-03-17 08:56:57

Visual StudVS2010C++

2010-01-25 16:58:15

C++程序

2010-03-10 09:28:58

Visual Stud

2010-03-17 08:49:49

Visual Stud

2019-11-06 14:45:09

網絡安全IT安全漏洞

2021-10-06 10:32:07

微軟Windows 11Windows

2010-03-22 16:38:31

Visual Stud

2009-09-02 16:21:17

Visual BasiC#語言

2019-11-04 05:38:45

網絡安全信息安全網絡攻擊

2021-10-04 14:59:35

微信手機安卓

2009-12-02 09:43:38

Visual Stud

2010-06-01 13:32:15

Visual Stud

2009-07-01 09:26:56

Visual Stud

2021-06-29 15:00:35

容器云紅帽

2010-01-14 10:56:43

Visual C++

2009-09-07 09:22:17

Visual Stud代碼片段

2010-03-19 13:17:26

Parallel

2009-11-24 09:00:02

Visual Stud

2010-07-15 08:50:09

SharePointVisual Stud

2010-07-20 08:43:00

Visual Stud
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 免费一看一级毛片 | 九九热这里 | 91久久夜色精品国产网站 | 亚洲一区二区三区视频免费观看 | 正在播放国产精品 | 国产高清精品一区二区三区 | 亚洲精品视频免费看 | 中文字幕视频在线观看 | 久久丁香 | 欧美一级电影免费 | 成人伊人 | 久久久久久久久久久久一区二区 | 色婷婷av久久久久久久 | 毛片一区二区 | 一区二区三区在线播放 | 国产成人精品一区二区 | 日韩一区二区三区精品 | 国产二区在线播放 | 亚洲精品久久久久中文字幕欢迎你 | 暖暖成人免费视频 | 91免费高清视频 | 欧美簧片| 欧美在线视频二区 | 久久亚洲精品视频 | 高清成人av | 欧美一区二区三区国产精品 | 中文天堂在线观看 | 国产精品久久久久久吹潮 | 一区二区在线免费观看视频 | h片在线观看免费 | 免费视频中文字幕 | 国产精品123区 | 亚洲91视频 | 国产精品a免费一区久久电影 | 国产伊人精品 | 精品成人免费视频 | 国产精品视频在线观看 | 成人性生交大片免费看中文带字幕 | 婷婷91| 日韩精品视频中文字幕 | 精品视频久久久 |