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

C++分析研究  
C++
日歷
<2013年2月>
272829303112
3456789
10111213141516
17181920212223
242526272812
3456789
統計
  • 隨筆 - 92
  • 文章 - 4
  • 評論 - 4
  • 引用 - 0

導航

常用鏈接

留言簿

隨筆檔案

文章檔案

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

 
一切緣自一位C語言開發經驗非常豐富的的朋友問我的一個問題。朋友問:“C++中的new在分配內存失敗時會拋出異常(std::bad_alloc)而不返回0(一些老的編譯器可能還在返回0,但這樣的編譯器實在”太老了“),這跟C程序員的做法很不一樣。而且,許多C++程序在使用new創建對象時也根本不檢查這種異常。這是一種什么哲學呢?”他還提到:“一般C程序員總會判斷一下malloc失敗的情況,就連Linux內核中都是如此。”

對于他的疑惑,我首先想到的是:一般用C++實現的應用層程序,內存管理方面自不能與內核程序相提并論。OS內核直接管理物理內存,所有應用程序的地址空間均由它映射而來,然后靠它建立機制進行翻譯。內核如果在內存管理方面不保險,應用層還怎么過日子?內核中的內存分配還須考慮許多其它問題,比如不同區域的不同特性(像某些DMA使用的buffer要物理連續且位于特定位置)。同樣重要的:對一個成熟的OS內核來說,即使在應用程序出現嚴重問題的時候也不能泄露物理內存及其它資源,且不能影響其它程序。

應用層程序則不同,它們一般擁有彼此獨立的、flat的虛擬內存空間,數量上通常遠大于物理內存。因此,一個應用程序如果能耗盡虛擬內存,那要么是對數據的規模估計不足,要么就是一個必需專門解決的嚴重bug.

耗盡虛擬內存跟其它許多嚴重的bug(再比如緩沖區溢出導致的堆棧破壞)一樣,多數情況下即使能檢測到也常常無計可施,如果“有計可施”,那何不早施此計?何苦等它發生再亡羊補牢呢?反過來想,該失敗的時候痛痛快快的快速失敗,這不算壞事。至少,比帶著問題繼續運行半小時,然后在某個完全不相干的地方發生莫名其妙又難以重現的bug要好得多。

這是我當時給朋友的回答,朋友勉強同意了,至少不再糾結C++程序員因何不在new的時候檢查std::bad_alloc了。然而,順著這個問題,我覺得可以聯想到好多相關的話題。

(1)首先想到的是Java語言的做法。Java中的變量都是引用(基本類型的除外),而被引用的對象是用new在堆(heap)上創建的。在Java中new一個對象時,理論上也有可能引發java.lang.OutOfMemoryError.當然,這是個Error,不是從java.lang.Exception派生的“異常”,因此語言并沒強制我們catch它。然而,語言是否要求并不重要,語言為什么不要求才是重要的。顯然,如果問題足夠嚴重,即使語言不要求,Java程序員也會在每一處new的周圍包上try/catch.可Java程序員沒有這么做。為什么?我想關鍵的原因跟上面是一樣的:一個應用程序耗盡虛擬內存,要么是對數據的規模估計不足(是否應通過java命令的-Xm系列參數設置更大的heap呢?),要么就是一個必須專門解決的bug www.yz-jx.com www.yz-jjx.com

同時,相對C++來說,Java程序中采用這一決策還有更充分的理由:因為有GC機制,Java程序中因為粗心造成的內存泄露較少(可能會有因不良設計造成的內存偽泄露)。

(2)C++中的“new”還不只是分配內存那么簡單。對于用戶自定義的類型來說,“new T;”相當于operator new再加上對T的構造函數的調用。由于類的構造函數完全可能引發異常,于是,就算內存分配一切順利,一條new語句還是可能產生異常。看來,需要catch的不止std::bad_alloc. www.js-yg.com  www.jx-liang.com

(3)暫不考慮“哲學”因素,如果有人仍然覺得應該像C程序那樣嚴格檢查內存分配,可不可以呢?當然可以,畢竟它還能拋出異常么,它能拋出我們就能捕捉。于是人們自然會想:C++或Java程序員用駝鳥策略對付內存分配的失敗,異常在使用上比較麻煩的事實會不會是原因之一呢?表面看是顯然的:每分配一次內存都要包上一層try/catch,跟C中的針對返回值的if/else風格比起來凌亂多了。

實際上,那不是使用異常的正確方法。如果異常只是if/else的簡單語法替代物,那它根本就沒有存在的必要。異常的好處之一(真的只是“之一”)是:一個異常只需一個地方處理就足夠了。比如下面這樣:

void f1() { try { // ……

f2();} catch (const some_exception& e) { // ……

}

void f2() { // ……

f3();}

void f3() { // ……

f4();}

void f4() { // ……

throw some_exception();}

[NextPage]

f4惹禍,f1收場,中間f2和f3只是一臉無辜地把異常“透過去”了(在Java中可能要聲明一下)——原因很可能是它們不具備足夠的上下文來處理這個異常。于是,我們不用像使用返回值那樣,從發生問題的地方開始,到處理問題的地方“之下”,中間每一層都要判斷一下,從而寫下一層又一層的諸如:

x = f();if(x < 0)

return x;之類的語句。你不覺得這樣可以使大多數函數更加干凈嗎?在異常或錯誤處理的問題上,這也使得不同邏輯層次的責任更加清晰。

值得一提的是,在異?;貪L的過程中,棧上已經構造好的對象都會正常析構。當然,這要求程序員在設計類的時候要考慮“異常安全”的因素。

關于異常處理的思想和異常的使用,完全可以講一本書。更有興趣的朋友不妨看看Herb Sutter寫的“Exceptional三卷本”:《Exceptional C++》、《More Exceptinal C++》和《Exceptional C++ Style》。

(4)事實上,C++中并非只有拋出異常的new,也有不拋異常的new,即通常所說的“nothrow new”。可以這樣使用它:

#include // ……

T* p = new (std::nothrow) T(/* …… */);其中,nothrow是頭文件中定義的一個類型為std::nothrow_t的常量,我們可以直接使用它。這時,如果內存分配失敗,p的值將為空(0),且不會有異常拋出,跟C的malloc很像了。

nothrow new實際是標準庫中實現的operator new和operator new[]的重載。我們也可以根據自己的需要重載operator new/operator new[],可以有全局的,也可以針對某個類重載。但實踐中用的不多。

注意,使用nothrow new創建對象時,只能保證不會因為operator new或operator new[]的失敗而拋出std::bad_alloc,但難保對象的構造函數不會拋出其它異常,甚至就拋出std::bad_alloc.

(5)說到C++的內存分配,還有必要提一下set_new_handler.它允許你設置一個在operator new和operator new[]分配內存失敗時可以回調的函數。如果你覺得在std::bad_alloc發生時還有什么辦法能改善一下內存使用的情況,這個回調函數也算得一根救命稻草托福代考 雅思代考

(6)雖然當std::bad_alloc發生時我們常常已無計可施,但并非所有的異常都如此,有些異常是可以處理從而挽回損失的。因此,在主函數最后,或者在多線程程序,尤其是所謂的worker thread的線程函數退出之前,用“catch(……)”捕捉一下所有異常還是有好處的。即使不指望恢復什么,至少不要因為一個線程而掛掉整個程序,同時盡量確保數據的完整性。

但別指望catch(……)能捕獲一切“問題”或“bug”,沒有那么好的事情。它只能捕獲C++的異常,其它的問題,比如前面提到的堆棧破壞,再比如野指針訪問,哪有那么容易檢測得到。(話說我最近被野指針搞得煩死了,不是我寫的。)

通常一個線程crash會導致整個進程crash,有人因為這個原因而更傾向于使用多進程,尤其是在類Unix的環境中。我個人對此雖不反對也不是特別贊同,因為欠債總是要還的,這也包括“技術債務”:有bug遲早還是要找到根源并真正解決,哪怕“真正繞開”也行托福答案 雅思答案

不過,使用多進程還有別的好處,因為進程間共享數據比同一個進程的線程之間要麻煩得多,這會迫使開發者做出減少共享,從而既能減少并發問題又能提高并發效率的設計。步入多核時代之后,讓并發實體盡可能地獨立,從而充分發揮硬件的并行性能,比什么都重要。

(7)我的另一個好朋友兼同事認為:程序crash沒有那么可怕。它可能是多數客戶最難以忍受的bug,但那只是源于社會心理,不見得是真正最嚴重的bug.

posted on 2013-02-27 15:55 HAOSOLA 閱讀(884) 評論(0)  編輯 收藏 引用
 
Copyright © HAOSOLA Powered by: 博客園 模板提供:滬江博客
PK10開獎 PK10開獎
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            美女免费视频一区| 国产精品久久午夜| 亚洲激情另类| 欧美成人午夜剧场免费观看| 欧美91福利在线观看| 亚洲精品日韩在线观看| 亚洲国产成人在线视频| 欧美精品在线免费| 亚洲综合99| 久久精品二区亚洲w码| 亚洲精品一区中文| 中国日韩欧美久久久久久久久| 国产精品啊啊啊| 久久一本综合频道| 欧美高清在线一区| 亚洲欧美日韩成人| 久久蜜臀精品av| 夜夜精品视频| 亚洲欧美精品在线观看| 在线看日韩av| 在线亚洲观看| 91久久久久久久久| 亚洲男人第一网站| 亚洲黄页视频免费观看| 亚洲在线一区二区| 亚洲国产高清自拍| 亚洲一级一区| 亚洲免费电影在线| 伊人婷婷久久| 日韩亚洲欧美在线观看| 亚洲特级毛片| 亚洲精品一区二区三区四区高清| 亚洲天堂成人在线观看| 亚洲国产精品va在线看黑人| 亚洲私人影院在线观看| 亚洲国产精品欧美一二99| 亚洲一区三区视频在线观看| 亚洲精品视频免费在线观看| 午夜精品婷婷| 亚洲一区在线视频| 美女999久久久精品视频| 亚洲欧美日韩在线播放| 欧美久久视频| 亚洲高清在线| 黑人一区二区| 亚洲自拍偷拍福利| 99伊人成综合| 欧美电影免费观看高清| 美女久久一区| 国产在线欧美日韩| 亚洲欧美资源在线| 亚洲无限av看| 欧美日韩在线观看一区二区| 亚洲电影在线看| 一区二区三区我不卡| 午夜在线观看免费一区| 午夜精品电影| 国产精品久久久久久久久久三级| 亚洲欧洲三级电影| 亚洲精品欧美专区| 欧美国产日本在线| 亚洲国产高潮在线观看| 亚洲国产精品黑人久久久| 久久精品国产久精国产思思| 欧美制服丝袜| 国产欧美一区二区三区沐欲 | 欧美成人国产va精品日本一级| 国产亚洲激情视频在线| 亚洲综合色在线| 久久精品日产第一区二区| 国产午夜精品理论片a级大结局| 亚洲一区三区视频在线观看| 性欧美暴力猛交69hd| 国产精品一二三四区| 午夜视频精品| 美女视频网站黄色亚洲| 亚洲激情在线| 欧美日韩裸体免费视频| 亚洲婷婷综合久久一本伊一区| 亚洲影院一区| 国产精品资源在线观看| 久久精品日韩欧美| 亚洲第一网站免费视频| 99精品国产99久久久久久福利| 欧美日韩亚洲国产一区| 亚洲欧美国产日韩天堂区| 久久精品99| 亚洲黄色影片| 国产精品免费区二区三区观看| 午夜精品久久久久久久| 亚洲第一福利在线观看| 亚洲一区二区毛片| 国内外成人在线| 亚洲视频观看| 亚洲黄一区二区| 久久综合色播五月| 亚洲激情一区二区| 亚洲男人的天堂在线观看 | 免费人成精品欧美精品| 99在线热播精品免费| 久久久99国产精品免费| 亚洲精品欧美专区| 国产在线拍偷自揄拍精品| 欧美高清视频一区| 欧美亚洲专区| 亚洲乱码国产乱码精品精98午夜| 西瓜成人精品人成网站| 亚洲国产精品女人久久久| 国产欧美精品xxxx另类| 欧美成人午夜激情| 午夜精品视频在线观看| 夜夜嗨一区二区| 欧美成人免费播放| 欧美一区二区啪啪| 一区二区av在线| 伊人夜夜躁av伊人久久| 国产精品爽爽ⅴa在线观看| 女同性一区二区三区人了人一| 亚洲欧美日韩国产成人| 日韩视频一区二区在线观看 | 老司机凹凸av亚洲导航| 亚洲字幕一区二区| 夜夜嗨一区二区三区| 欧美 日韩 国产在线| 久久aⅴ国产欧美74aaa| 一区二区久久久久| 亚洲国产小视频| 有码中文亚洲精品| 国产综合久久久久影院| 国产精品国产三级国产普通话蜜臀 | 久久久久国产精品人| 亚洲欧美日韩综合国产aⅴ| 一区二区三区四区蜜桃| 最新日韩在线视频| 欧美大片第1页| 欧美成人免费观看| 欧美91大片| 欧美成人资源| 欧美肥婆在线| 欧美激情在线| 亚洲国产成人91精品| 欧美国产精品劲爆| 欧美激情四色| 亚洲国产综合在线看不卡| 欧美成人免费全部| 亚洲电影第1页| 亚洲精品乱码久久久久久| 亚洲国产日韩欧美在线动漫| 欧美黄色影院| 99国产精品久久久久老师| 一区二区激情| 午夜精品久久久久久久99黑人| 欧美一区二区久久久| 久久九九电影| 女女同性精品视频| 欧美日韩成人在线观看| 国产精品女主播| 国产日韩在线播放| 亚洲成人在线| 一区二区三区黄色| 欧美一区二区三区免费看 | 欧美日韩成人在线视频| 国产精品二区三区四区| 国产亚洲综合精品| 亚洲精品国偷自产在线99热| 中文日韩在线视频| 久久精品123| 亚洲人成人77777线观看| 亚洲午夜一区二区| 久久久精品国产免大香伊| 欧美大片va欧美在线播放| 欧美精品在线观看91| 国产精品福利在线观看网址| 国产一区亚洲| 亚洲精品久久久久久久久久久久| 亚洲精品日韩一| 久久av红桃一区二区小说| 久久婷婷蜜乳一本欲蜜臀| 欧美激情一区二区三区在线| 亚洲精品欧美日韩专区| 亚洲视频你懂的| 欧美诱惑福利视频| 美国十次成人| 欧美日韩精品免费观看| 欧美精品一区在线观看| 国产乱码精品一区二区三区忘忧草 | 欧美国产欧美亚洲国产日韩mv天天看完整 | 久久久久综合一区二区三区| 久久免费视频网| 亚洲第一免费播放区| 亚洲欧洲一区二区三区久久| 欧美一区日韩一区| 欧美国内亚洲| 国产精一区二区三区| 亚洲网站在线| 蜜臀久久99精品久久久久久9 | 亚洲人人精品| 午夜一级久久| 一本色道久久综合亚洲二区三区| 亚洲一区二区三区精品在线观看|