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

C小加

厚德 博學 求真 至善 The bright moon and breeze
posts - 145, comments - 195, trackbacks - 0, articles - 0
  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

現代C++風格的新元素

Posted on 2011-11-14 13:07 C小加 閱讀(556) 評論(0)  編輯 收藏 引用 所屬分類: C++和面向對象

“C++11就像一門新的語言。” – Bjarne Stroustrup

C++11標準推出了很多有用的新特性。本文特別關注相比C++98更像是一門新語言的那部分特性,因為:

這些特性改變了編寫C++程序使用的代碼風格和習語[TODO],通常也包括你設計C++函數庫的方式。例如,你會看到更多參數和返回值類型為智能指針(smart pointer),同時也會看到函數通過值傳遞返回大型對象

你將會發現在大多數的代碼示例中充斥著新特性的身影。例如,幾乎每5行現代C++代碼示例都會使用到auto

C++11的其他特性也很棒。但是請先熟悉下面這些,正是因為這些特性的廣泛使用使得C++11代碼如同其他現代主流語言一樣整潔、安全和高效,與此同時保持了C++傳統的性能優勢。

提示:

  • 與Strunk & White[TODO]一樣,本文只做概要的總結性指導而不做詳盡基本原理和優缺點分析。詳細分析請參見其他文章
  • 本文會不斷更新,主要變更及內容增加請參見文末變更記錄

auto

nullptr

始終使用nullptr表示一個null指針值,絕不要使用數字0或者NULL宏,因為它們也可以代表一個整數或者指針從而產生歧義。

Range for

基于范圍的循環使得按順序訪問其中的每個元素變得非常方便。

基于以下兩個原因,盡可能使用auto:首先,使用auto會避免重復聲明編譯器已經知道的類型。 
 
  
// C++98
  map<int,string>::iterator i = m.begin ();

  
// C++11

  auto i 
= begin (m);

其次,當使用未知類型或者類型名稱不易理解時使auto會更加便利,例如大多數的lambda函數[TODO]——你甚至不能簡單的拼寫出類型的名字。

 
// C++98
  binder2nd< greater<int> > x = bind2nd ( greater<int>(), 42 );

  
// C++11

  auto x 
= [](int i) { return i > 42; };

需要注意,使用auto并不改變代碼的含義。代碼仍然是靜態類型[譯注],每個表達式的類型都是清晰和明確的;C++11只是不需要我們重復聲明類型的名字。一些人剛開始可能會害怕在這里使用auto,因為感覺好像沒有(重復)聲明我們需要的類型就意味著會碰巧得到一個不同的類型。如果你想要明確地進行一次強制類型轉換,沒有問題,聲明目標類型就好了。然而大多數情況下,只要使用auto就可以了;幾乎不會出現錯誤地拿到一個不同類型的情況,即便出現錯誤,C++的強靜態類型系統也會由編譯器讓你知道這個錯誤,因為你正試圖訪問一個變量沒有的成員函數或是錯誤地調用了該函數。

譯注:動態類型語言(dynamic typing language)是指類型檢查發生在運行期間(run-time)的語言。靜態類型語言(static typing language)是類型檢查發生在編譯期間(compile-time)的語言。

智能指針:無須delete

請始終使用標準智能指針以及非占有原始指針(non-owning raw pointer)。絕不要使用占有原生指針(owning raw pointer)和delete操作,除非在實現你自己的底層數據結構這種少見的情況下(即使在此時也需要在 class 范圍內保持完好的封裝)。如果只能夠知道你是另一個對象唯一的所有者,請使用unique_ptr來表示唯一所有權(TODO)。一個”new T”表達式會馬上初始化另一個引用它的對象,通常是一個unique_ptr。



 
// C++11 Pimpl Idiom
  class widget {

  widget ();

  
~widget ();

  
private:

  
class impl;

  unique_ptr
<impl> pimpl;

  };

  
// in .cpp file

  
class impl {

  :::

  };

  widget::widget ()

  : pimpl ( 
new impl () )

  {

  }

  widget::
~widget () = default;
使用shared_ptr來表示共享所有權。推薦使用make_shared來有效地創建共享對象。

 
// C++98
  widget* pw = new widget ();

  :::

  delete pw;

  
// C++11

  auto pw 
= make_shared<widget>();

使用 weak_ptr 來退出循環并且表示可選性(例如,實現一個對象緩存)

  
// C++11
  class gadget;

  
class widget {

  
private:

  shared_ptr
<gadget> g; // if shared ownership

  };

  
class gadget {

  
private:

  weak_ptr
<widget> w;

  };

如果你知道另一個對象存在時間會更長久并且希望跟蹤它,使用一個(非占有non-owning)原始指針。

  
// C++11
  class node {

  vector
< unique_ptr<node> > children;

  node
* parent;

  
public:

  :::

  };

nullptr

始終使用nullptr表示一個null指針值,絕不要使用數字0或者NULL宏,因為它們也可以代表一個整數或者指針從而產生歧義。

Range for

基于范圍的循環使得按順序訪問其中的每個元素變得非常方便。


  
// C++98
  for( vector<double>::iterator i = v.begin (); i != v.end (); ++i ) {

  total 
+= *i;

  }

  
// C++11

  
for( auto d : v ) {

  total 
+= d;

  }

非成員(nonmember) begin 和 end

始終使用非成員begin和end,因為它是可擴展的并且可以應用在所有的容器類型(container type),不僅僅是遵循了STL風格提供了 .begin ()和 .end ()成員函數的容器,甚至數組都可以使用。

如果你使用了一個非STL風格的collection類型,雖然提供了迭代但沒有提供STL的 .begin ()和 .end (),通常可以為這個類型編寫自己的非成員begin和end來進行重載。這樣你就可以使用STL容器的編程風格來遍歷該類型。C++11標準提供了示例:C數組就是這樣一個類型,標準同時為數組提供了begin和end。


  vector
<int> v;
  
int a[100];

  
// C++98

  sort ( v.begin (), v.end () );

  sort ( 
&a[0], &a[0+ sizeof(a)/sizeof(a[0]) );

  
// C++11

  sort ( begin (v), end (v) );

  sort ( begin (a), end (a) );

Lambda 函數和算法

Lambda[TODO]是決定乾坤的因素,它會使你編寫的代碼變得更優雅、更快速。Lambda使得STL算法的可用性提高了近100倍。新近開發的C++函數庫都是基于lambda可以用的前提(例如,PPL)并且有些函數庫甚至要求你編寫lambda來調用函數庫(例如,C++ AMP)

下面是一個快速示例:找到v里面大于x并且小于y的第一個元素。在C++11中,最簡單和干凈的代碼就是調用一個標準函數。


  
// C++98: write a naked loop (using std::find_if is impractically difficult)
  vector<int>::iterator i = v.begin (); // because we need to use i later

  
for( ; i != v.end (); ++i ) {

  
if*> x && *< y ) break;

  }

  
// C++11: use std::find_if

  auto i 
= find_if ( begin (v), end (v), [=](int i) { return i > x && i < y; } );

想要使用C++編寫一個循環或者類似的新特性?不用著急;只要編寫一個模板函數(template function)(函數庫算法),并且幾乎可以將lambda當做語言特性來使用,與此同時會更加靈活,因為它不是固定的語言特性而是一個真正的函數庫。

  
// C#
  lock( mut_x ) {

   use x 

  }

  
// 不使用 lambda 的C++11:已經非常簡潔并且更靈活(例如,可以使用超時以及其他選項)

  {

  lock_guard
<mutex> hold ( mut_x );

   use x 

  }

  
// C++11 with lambdas, and a helper algorithm: C# syntax in C++

  
// 使用了 lambda 的C++11可以帶一個輔助算法:在 C++ 中使用 C# 的文法

  
// Algorithm: template<typename T, typename F> void lock ( T& t, F f ) { lock_guard<T> hold (t); f (); }

  
lock( mut_x, [&]{

   use x 

  });

去熟悉lambda吧。你會大量使用它,不僅僅在C++中——它已經廣泛應用于很多主流的編程語言。一個開始的好去處請參考我在PDC2010的演講《無處不在的 lambda》

Move / &&

Move被認為是copy的最佳優化,盡管它也使得其他事情成為可能比如信息被轉發。



// C++98:避免 copy 的替代方法
  vector<int>* make_big_vector (); // 選擇1: 返回指針: 沒有拷貝,但不要忘記 delete

  :::

  vector
<int>* result = make_big_vector ();

  
void make_big_vector ( vector<int>& out ); // 選擇2: 通過引用傳遞: 沒有拷貝,但是調用者需要傳入一個有名對象

  :::

  vector
<int> result;

  make_big_vector ( result );

  
// C++11: move

  vector
<int> make_big_vector (); // 通常對于”被調用者(callee)分配的空間“也適用

  :::

  vector
<int> result = make_big_vector ();

Move語法改變了我們設計API的方式。我們可以更多地設計通過值傳遞。為你的類型啟用move語法,使用時會比copy更有效。

更多變化

還有更多現代C++的特性。并且我計劃在未來編寫更多深入C++11新特性以及其他特性的短文,我們會知道更多并且喜歡上它。

但目前,這是必須知道的新特性。這些特性組成了現代C++風格的核心,使得C++代碼看起來和執行時像他們設計的那樣,你將會看到這些特性會出現在幾乎每一段你看到或者編寫的現代C++代碼中。并且它們使得現代C++更加干凈、安全且快速,使得C++在未來的若干年仍然是我們產業的依靠。

主要變更

2011-10-30:為Lambda增加C#lock示例。重新組織智能指針首先介紹unique_prt。

文章出自:伯樂在線







 

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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精品视频免费全部在线| 亚洲专区一区二区三区| 亚洲激情啪啪| 亚洲欧美日本视频在线观看| 亚洲激情不卡| 亚洲一区欧美一区| 亚洲欧洲日韩女同| 午夜在线a亚洲v天堂网2018| 亚洲国产一区二区视频| 亚洲视频图片小说| 亚洲精品国精品久久99热一| 亚洲一区自拍| 一本久久综合| 快she精品国产999| 欧美在线观看视频在线| 欧美高清在线一区| 久久这里只精品最新地址| 国产精品久久久久久久久久直播 | 欧美激情精品久久久久久变态| 国产精品久久久久一区二区| 亚洲国产精品一区二区www| 国产亚洲福利一区| 一区二区日韩伦理片| 亚洲精品国产精品乱码不99| 亚洲欧美激情诱惑| 中文国产成人精品久久一| 欧美在线观看一区二区| 亚洲男人的天堂在线aⅴ视频| 久久综合九色综合欧美狠狠| 久久久久久一区| 国产模特精品视频久久久久 | 欧美阿v一级看视频| 国产欧美日韩综合精品二区| 99精品黄色片免费大全| 亚洲精品视频在线| 美女视频黄免费的久久| 老色鬼精品视频在线观看播放 | 久久久久久一区二区| 久久精品视频99| 国产精品亚洲精品| 亚洲午夜一二三区视频| 亚洲自拍都市欧美小说| 欧美午夜片在线观看| 99精品国产热久久91蜜凸| 99re6这里只有精品| 欧美区高清在线| 亚洲免费大片| 亚洲一区二区三区乱码aⅴ| 欧美日韩精品欧美日韩精品一| 亚洲国产一区在线| 99综合电影在线视频| 欧美日韩精品中文字幕| 一级成人国产| 羞羞漫画18久久大片| 国产欧美日韩伦理| 欧美中文字幕在线视频| 蜜臀99久久精品久久久久久软件| 亚洲国产成人精品久久| 欧美freesex交免费视频| 亚洲精品免费网站| 亚洲在线电影| 国产亚洲午夜| 欧美成人精品1314www| 亚洲欧洲日本在线| 亚洲天堂免费观看| 国产日韩一区欧美| 久久天天躁夜夜躁狠狠躁2022| 欧美激情亚洲国产| 亚洲影视在线| 红杏aⅴ成人免费视频| 欧美大片在线看| 国产精品99久久不卡二区| 久久九九久精品国产免费直播| **欧美日韩vr在线| 欧美日韩国产综合视频在线观看 | 欧美jjzz| 亚洲一区在线直播| 激情欧美一区二区三区| 欧美激情成人在线视频| 亚洲女人天堂av| 亚洲福利av| 午夜激情综合网| 亚洲激情亚洲| 国产欧美日韩一级| 欧美激情视频一区二区三区免费| 亚洲女优在线| 亚洲日本电影在线| 久久夜色精品国产噜噜av| 艳女tv在线观看国产一区| 国产亚洲欧美日韩美女| 午夜精品理论片| 免费欧美网站| 日韩一级精品视频在线观看| 美女视频一区免费观看| 亚洲调教视频在线观看| 亚洲高清色综合| 国产日韩欧美高清免费| 欧美三级网址| 欧美v国产在线一区二区三区| 午夜欧美电影在线观看| 日韩亚洲在线观看| 亚洲电影激情视频网站| 久久天天躁夜夜躁狠狠躁2022| 亚洲欧美伊人| 在线亚洲观看| 亚洲另类在线一区| 亚洲国产一二三| 黄色日韩网站| 国模大胆一区二区三区| 国产美女精品一区二区三区| 欧美性事免费在线观看| 欧美日韩另类视频| 欧美精品一区二区三区四区| 美女黄毛**国产精品啪啪| 久久精品国产久精国产思思| 香蕉久久a毛片| 欧美亚洲一区二区在线| 亚洲免费影视| 午夜免费在线观看精品视频| 亚洲综合第一页| 亚洲欧美另类国产| 亚洲欧美日韩人成在线播放| 亚洲一区二区视频在线| 亚洲综合色噜噜狠狠| 亚洲欧美在线免费观看| 性色一区二区三区| 欧美在线一区二区| 欧美伊人影院| 久久夜色精品国产欧美乱极品| 久久天天躁狠狠躁夜夜av| 久久综合狠狠综合久久激情| 久久久久一区二区| 免费日韩成人| 欧美绝品在线观看成人午夜影视| 欧美日韩精品综合在线| 国产精品久久久久毛片软件 | 99亚洲伊人久久精品影院红桃| 日韩一级视频免费观看在线| 一区二区三区产品免费精品久久75| 在线中文字幕日韩| 欧美专区在线观看| 牛牛精品成人免费视频| 亚洲黄色成人| 亚洲小说欧美另类社区| 久久国产精品久久久久久久久久| 久久全球大尺度高清视频| 欧美激情第五页| 国产精品综合久久久| 激情五月***国产精品| 日韩视频二区| 久久爱www久久做| 欧美高清视频| 亚洲一区久久| 美女被久久久| 国产精品乱码| 亚洲高清网站| 性欧美暴力猛交69hd| 免费观看30秒视频久久| 日韩一级精品| 久久久噜噜噜久久人人看| 欧美日韩一区二区在线视频| 国产午夜久久久久| 亚洲视频精品在线| 久久免费视频这里只有精品| 日韩视频在线播放| 久久久久久久久久久久久久一区| 欧美日韩亚洲国产一区| 在线电影国产精品| 久久亚洲综合网| 亚洲综合日韩| 亚洲一二三区视频在线观看| 久久久www| 一本大道久久a久久精二百| 久久精品国产一区二区三区免费看| 欧美日韩视频在线| 亚洲第一色中文字幕| 欧美在线观看网址综合| 亚洲精品一区二区三区四区高清| 久久精品视频99| 国产伦精品一区二区三区视频黑人 | 99精品免费| 欧美成人首页| 久久精品免费看| 国产乱码精品一区二区三区五月婷|