青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆-381  評論-37  文章-0  trackbacks-0

轉:http://blog.csdn.net/yincheng01/archive/2010/03/11/5367032.aspx

微軟即將在2010年4月12日發布VS2010的正式版,對于c++語言做了修改,使之更加符合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表達式的學習難度,避免了使用復雜的函數對象或是函數指針所帶來的錯誤。我們可以看一個實際的例子:

view plaincopy to clipboardprint?
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
#include "stdafx.h"  
#include <algorithm>  
#include <iostream>  
#include <ostream>  
#include <vector>  
 
using namespace std;  
 
int _tmain(int argc, _TCHAR* argv[])  
{  
    vector<int> v;  
    for (int i = 0; i < 10; ++i) {  
        v.push_back(i);  
    }  
         for_each(v.begin(), v.end(), [] (int n) {  
        cout << n;  
        if (n % 2 == 0) {  
            cout << " even ";  
        } else {  
            cout << " odd ";  
        }  
    });  
    cout << endl;  
 
    return 0;  

#include "stdafx.h"
#include <algorithm>
#include <iostream>
#include <ostream>
#include <vector>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    vector<int> v;
    for (int i = 0; i < 10; ++i) {
        v.push_back(i);
    }
         for_each(v.begin(), v.end(), [] (int n) {
        cout << n;
        if (n % 2 == 0) {
            cout << " even ";
        } else {
            cout << " odd ";
        }
    });
    cout << endl;

    return 0;
}
 

  這段代碼循環遍歷輸出vector中的每一個數,并判斷這個數是奇數還是偶數。我們可以隨時修改Lambda表達式而改變這個匿名函數的實現,修改對集合的操作。在這段代碼中,C++使用一對中括號“[]”來表示Lambda表達式的開始,其后的”(int n)”表示Lambda表達式的參數。這些參數將在Lambda表達式中使用到。為了體會Lambda表達式的簡潔,我們來看看同樣的功能,如何使用函數對象實現:
 
view plaincopy to clipboardprint?
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
#include "stdafx.h"  
#include <algorithm>  
#include <iostream>  
#include <ostream>  
#include <vector>  
using namespace std;  
 
struct LambdaFunctor {  
    void operator()(int n) const {  
        cout << n << " ";  
        if (n % 2 == 0) {  
            cout << " even ";  
        } else {  
            cout << " odd ";  
        }  
 
    }  
};  
 
int _tmain(int argc, _TCHAR* argv[])  
{  
    vector<int> v;  
 
    for (int i = 0; i < 10; ++i) {  
        v.push_back(i);  
    }  
 
    for_each(v.begin(), v.end(), LambdaFunctor());  
    cout << endl;  
 
    return 0;  

#include "stdafx.h"
#include <algorithm>
#include <iostream>
#include <ostream>
#include <vector>
using namespace std;

struct LambdaFunctor {
    void operator()(int n) const {
        cout << n << " ";
        if (n % 2 == 0) {
            cout << " even ";
        } else {
            cout << " odd ";
        }

    }
};

int _tmain(int argc, _TCHAR* argv[])
{
    vector<int> v;

    for (int i = 0; i < 10; ++i) {
        v.push_back(i);
    }

    for_each(v.begin(), v.end(), LambdaFunctor());
    cout << endl;

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

靜態斷言static_assert

  在之前的C++標準C++03中,我們可以使用兩種斷言:
  • 使用預處理中的條件編譯和#error指令,可以在預處理階段檢查一些編譯條件
  • 可以使用宏assert來進行運行時檢查,以確保程序邏輯的正確性

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

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

view plaincopy to clipboardprint?
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
template <int N> struct Kitten {  
    static_assert(N < 2, "Kitten<N> requires N < 2.");  
};  
 
int main() {  
    Kitten<1> peppermint;  
    Kitten<3> jazz;  
 
   return 0;  

template <int N> struct Kitten {
    static_assert(N < 2, "Kitten<N> requires N < 2.");
};

int main() {
    Kitten<1> peppermint;
    Kitten<3> jazz;

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


圖2 static_assert斷言及其輸出


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

  auto關鍵字

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

view plaincopy to clipboardprint?
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
#include <iostream>  
#include <map>  
#include <ostream>  
#include <regex>  
#include <string>  
using namespace std;  
using namespace std::tr1;  
 
int main() {  
    map<string, string> m;  
 
    const regex r("(\\w+) (\\w+)");  
 
    for (string s; getline(cin, s); ) {  
        smatch results;  
        if (regex_match(s, results, r)) {  
            m[results[1]] = results[2];  
        }  
    }  
    for (auto i = m.begin(); i != m.end(); ++i) {  
        cout << i->second << " are " << i->first << endl;  
    }  
 
    return 0;  

#include <iostream>
#include <map>
#include <ostream>
#include <regex>
#include <string>
using namespace std;
using namespace std::tr1;

int main() {
    map<string, string> m;

    const regex r("(\\w+) (\\w+)");

    for (string s; getline(cin, s); ) {
        smatch results;
        if (regex_match(s, results, r)) {
            m[results[1]] = results[2];
        }
    }
    for (auto i = m.begin(); i != m.end(); ++i) {
        cout << i->second << " are " << i->first << endl;
    }

    return 0;
}
 
  在這段代碼中,我們使用auto關鍵字來代替了真正的數據類型map<string, string>::iterator,這使得整個代碼自然而簡潔。

  另外,跟其他數據類型一樣,我們也可以對auto關鍵字進行修飾,例如添加const,指針(*),左值引用(&),右值引用(&&)等等,編譯器會根據auto類型所代表的真正的數據來決定這些修飾的具體含義。

  為了兼容一些舊有的C++代碼,我們可以使用/Zc:auto這個編譯器選項,來告訴編譯器是采用auto關鍵字的原有定義還是在新標準C++0x中的定義。

右值引用

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

int a;
int& a_lvref = a;  // 左值引用

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

int& a_lvref = int();      // error C2440: 'initializing' : cannot convert from 'int' to 'int &'   
int&& b_rvref = int();  // OK!
  在第一行代碼中,我們將一個臨時對象int()綁定到一個左值引用,將產生一個編譯錯誤。而在第二行中,我們將臨時對象綁定到右值引用,就可以順利通過編譯。

  右值是無名的數據,例如函數的返回值一般說來就是右值。當對右值進行操作的時候,右值本身往往沒有必要保留,因此在某些情況下可以直接“移動”之。通過右值引用,程序可以明確的區分出傳入的參數是否為右值,從而避免了不必要的拷貝,程序的效率也就得到了提高。我們考慮一個簡單的數據交換的小程序,從中來體會右值引用所帶來的效率提升。我們可以寫一個函數swap來實現兩個變量值的交換:

view plaincopy to clipboardprint?
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
template <class T> swap(T& a, T& b)  
{  
    T tmp(a);   // tmp對象創建后,我們就擁有了a的兩份拷貝  
    a = b;      // 現在我們擁有b的兩份拷貝  
    b = tmp;    // 現在我們擁有a的兩份拷貝  

template <class T> swap(T& a, T& b)
{
    T tmp(a);   // tmp對象創建后,我們就擁有了a的兩份拷貝
    a = b;      // 現在我們擁有b的兩份拷貝
    b = tmp;    // 現在我們擁有a的兩份拷貝
}
  在這段代碼中,雖然我們只是為了進行簡單的數據交換,但是卻執行了多次對象拷貝。這些對象的拷貝操作,特別是當這些對象比較大的時候,無疑會影響程序的效率。

  那么,如果使用右值引用如何實現呢?

view plaincopy to clipboardprint?
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
#include "stdafx.h"  
 
template <class T>   
T&& move(T&& a)  
{  
    return a;  
}  
 
template <class T> void swap(T& a, T& b)  
{  
    T tmp(move(a)); // 對象a被移動到對象tmp,a被清空  
    a = move(b);    // 對象b被移動到對象a,b被清空  
    b = move(tmp);  // 對象tmp被移動到對象b  
}  
 
int _tmain(int argc, _TCHAR* argv[])  
{  
    int a = 1;  
    int b = 2;  
    swap(a, b);  
 
   return 0;  

#include "stdafx.h"

template <class T>
T&& move(T&& a)
{
    return a;
}

template <class T> void swap(T& a, T& b)
{
    T tmp(move(a)); // 對象a被移動到對象tmp,a被清空
    a = move(b);    // 對象b被移動到對象a,b被清空
    b = move(tmp);  // 對象tmp被移動到對象b
}

int _tmain(int argc, _TCHAR* argv[])
{
    int a = 1;
    int b = 2;
    swap(a, b);

   return 0;
}
 


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

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

  在Visual Studio 2010中,因為有了對這些C++0x新特性的支持,重新點燃了程序員們對C++的熱情。C++重振雄風,指日可待!


本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/yincheng01/archive/2010/03/11/5367032.aspx

 

posted on 2010-03-15 18:25 小王 閱讀(541) 評論(0)  編輯 收藏 引用 所屬分類: c++ 程序設計基礎
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲国产高清一区二区三区| 亚洲视频在线观看| 欧美成人黄色小视频| 久久久在线视频| 麻豆九一精品爱看视频在线观看免费| 欧美日韩视频不卡| 欧美亚洲自偷自偷| 欧美网站在线观看| 欧美性大战久久久久久久| 国产精品日韩欧美大师| 国产精品一区二区三区乱码| 国产欧美日韩高清| 亚洲国产精品美女| 一区二区三区日韩在线观看| 欧美在线观看视频一区二区| 久久精品噜噜噜成人av农村| 欧美高清在线一区| 亚洲一区二区免费| 久久中文字幕一区| 国产精品成人久久久久| 狠狠色狠狠色综合人人| 日韩视频不卡中文| 久久精品网址| 亚洲乱码国产乱码精品精可以看 | 性欧美激情精品| 久久精品日韩| 欧美日韩少妇| 亚洲国产欧美一区二区三区久久| 99国产一区| 久久天天狠狠| 欧美日韩国产精品自在自线| 国产精品久久久久久久久久妞妞 | 欧美日韩中文在线| 国产综合精品| 亚洲综合第一页| 欧美成人免费网站| 亚洲在线中文字幕| 欧美日韩一区在线观看| 在线成人激情| 欧美综合激情网| 亚洲精品免费网站| 久久久久一区二区三区四区| 国产精品女同互慰在线看| 亚洲精品五月天| 久久久精品国产免费观看同学| 在线视频日韩| 欧美成人免费全部| 久久精品亚洲乱码伦伦中文 | 欧美v亚洲v综合ⅴ国产v| 国产一区二区三区日韩欧美| 亚洲欧美日韩国产综合精品二区| 亚洲全部视频| 欧美激情一区二区三区蜜桃视频| 在线精品福利| 欧美超级免费视 在线| 久久亚洲风情| 一区二区三区在线高清| 久久精品久久综合| 欧美资源在线观看| 国产色综合久久| 久久精品国产清高在天天线| 欧美一区二区三区四区视频| 国产午夜精品久久| 久久久久**毛片大全| 欧美一区二区福利在线| 国内精品福利| 久久综合狠狠综合久久综合88| 亚洲欧美一区二区精品久久久| 国产欧美视频一区二区三区| 欧美一区二区三区四区在线观看地址 | 美女主播一区| 久久亚洲图片| 亚洲人成亚洲人成在线观看图片| 亚洲国产高清aⅴ视频| 欧美精品久久久久a| 中日韩美女免费视频网站在线观看| 亚洲免费观看| 国产精品一区二区男女羞羞无遮挡 | 99精品欧美一区二区蜜桃免费| 亚洲国产精品视频一区| 欧美另类极品videosbest最新版本 | 性久久久久久久久| 黄色国产精品| 亚洲欧洲另类| 国产精品一区二区久久| 免费观看久久久4p| 欧美日韩国产精品一区| 欧美在线综合| 欧美69wwwcom| 先锋影音网一区二区| 久久不见久久见免费视频1| 亚洲高清一区二区三区| 日韩午夜电影| 激情综合激情| 亚洲视频高清| 亚洲经典在线看| 亚洲一区二区三区在线观看视频| 激情成人亚洲| 在线综合欧美| 亚洲国产毛片完整版| 亚洲网在线观看| 亚洲狠狠婷婷| 欧美一区久久| 亚洲在线视频网站| 国产精品久久午夜夜伦鲁鲁| 久久亚洲色图| 欧美日韩综合在线免费观看| 久久综合九色欧美综合狠狠| 欧美四级电影网站| 欧美夫妇交换俱乐部在线观看| 国产精品红桃| 91久久在线| 在线免费观看欧美| 香港久久久电影| 亚洲欧美激情在线视频| 欧美国产欧美亚洲国产日韩mv天天看完整| 午夜精彩视频在线观看不卡 | 欧美日韩国产色综合一二三四| 久久精品欧洲| 国产精品欧美风情| 99精品视频一区二区三区| 尹人成人综合网| 亚洲欧美综合| 欧美一区二区高清| 国产精品久久久久久久9999| 亚洲精品韩国| 亚洲美女黄网| 欧美国产极速在线| 亚洲国产精品嫩草影院| 亚洲激情亚洲| 欧美成人一区二区| 亚洲高清成人| 亚洲乱码视频| 欧美精品久久久久久久久老牛影院| 免播放器亚洲一区| 在线看日韩av| 麻豆精品视频| 亚洲福利免费| aa级大片欧美三级| 欧美日韩综合在线免费观看| 亚洲狼人综合| 香蕉尹人综合在线观看| 国产精品成人观看视频免费 | 欧美激情一区二区三区在线视频观看| 猫咪成人在线观看| 老司机免费视频一区二区| 久久久久久久久久久久久女国产乱| 国产精品亚洲综合天堂夜夜| 欧美一级久久久| 欧美不卡视频| 中文精品99久久国产香蕉| 国产精品久久7| 欧美一区二区私人影院日本| 麻豆精品91| 一本一本a久久| 国产精品久久久久高潮| 久久er精品视频| 欧美激情第8页| 亚洲午夜av在线| 国产亚洲精品美女| 免费精品99久久国产综合精品| 亚洲精品自在在线观看| 午夜综合激情| 亚洲国产高清一区二区三区| 欧美理论在线| 欧美在线电影| 日韩亚洲在线观看| 久久久综合香蕉尹人综合网| 亚洲精品视频免费| 国产亚洲第一区| 欧美高清在线视频观看不卡| 欧美高清一区| 欧美激情视频在线播放| 蜜臀久久久99精品久久久久久| 欧美三区美女| 亚洲精品午夜| 欧美激情亚洲综合一区| 亚洲美女在线观看| 欧美91大片| 亚洲综合二区| 国产亚洲网站| 亚洲女优在线| 一区二区三区导航| 国产噜噜噜噜噜久久久久久久久| 亚洲伊人网站| 美腿丝袜亚洲色图| 亚洲欧美日韩人成在线播放| 影音先锋日韩精品| 国产精品亚洲第一区在线暖暖韩国| 久久综合电影一区| 亚洲一区二区三区欧美| 亚洲国产精品成人精品 | 夜夜嗨av一区二区三区中文字幕| 国产欧美日韩免费看aⅴ视频| 欧美xx69| 欧美大片91| 老妇喷水一区二区三区| 欧美一区=区| 亚洲一区二区三区在线看| 亚洲精品一区二区三区蜜桃久 |