• <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>
            隨筆-90  評論-947  文章-0  trackbacks-0

            先看一個例子。首先,我要寫一個vector;其次,為了使用方便,我需要提供一個帶 size 參數的構造函數。要求就這兩點。

            那么,勢必要:

            class vector
            {
            public:
                vector(size_t size)
                {
                    // ...
                    m_pData = new int[size]; // 假設就是 int 這樣的基本類型好了,以避免下面可能出現的離題
                    // ...
                }
            };

            問題來了。new 不是有可能失敗嗎?失敗了在老編譯器里會返回 NULL(這個情形也先無視),在新編譯器里會拋異常。那么,在這里要不要進行檢查呢?如果檢查:

            try
            {
                m_pData = new int[size];
            }
            catch (...)
            {

            }

            catch到了。那么在這里可以干啥呢?似乎。。。啥也干不了!作為構造函數,沒法使用返回值,自然只能使用異常來提示外界;既然本來就是異常,我又何必在這里 try 一次呢?(假設這里沒有其他錯誤要處理,也假設這里的類型是int之類的基本類型,不會出現執行元素的構造函數失敗的情形)

            既然這里的 try 讓我們如此無奈,那么就不必 try 了。這個時候,我需要給 vector(size_t size) 標記上 throw 嗎?如果不標記,使用者怎么知道這里可能會有異常?如果標記了,或者沒標記但使用者意識到了,那么他會這樣用:

            try
            {
                vector v(10);
                // Task with v
                // ...
                // ...
                // ...
            }
            catch (...)
            {
                // Error handler
            }

            因為 v 的作用域被限制在了 try 內,所以所有的與 v 相關的邏輯代碼全部要放在 try 內部了。這種樣子似乎與 C# 很像!在 C# 里,try...catch... 是標準的做法;但是在 C++ 里,似乎不會如此經常地用 try catch,要不然,為什么我見過的 C++ 代碼都不是這樣子的呢?兩年前在金山實習的時候,有一次我把 try...catch 當做通用的錯誤處理來做,所有的錯誤都搞成一種異常,返回值僅返回正常值。結果董波叔叔說,這樣子是不對滴,但是沒給出讓我信服理由,可能就是,C++ 的 try...catch 的性能很不好之類的。(C# 以及 Java 的 try...catch 的性能好嗎?)

            好,既然大家都不這么辦,是不是這里也不用 try 了?于是,內存分配錯誤就讓它自生自滅了……記得以前某本書上看到,說這種情形下的處理,僅僅是一個道德問題而已。真的無解嗎?

            如果放寬要求,不要求在構造函數提供內存分配,那倒是有一種解法——分兩階段構造:

            class vector
            {
            public:
                vector()
                {
                    // ...
                }
                bool allocate(size_t size)
                {
                    try
                    {
                        m_pData = new int[size];
                    }
                    catch (...)
                    {
                        return false;
                    }
                    if (m_pData == NULL)
                    {
                        return false;
                    }
                    // Other code ...
                    return true;
                }
            };

            但是使用起來就不“方便”了。現實中,這種情形倒是存在,如 CWindow 的 Create,還有啥啥啥的 Init 等等。

            真的沒有辦法兼顧方便與安全嗎?

            posted on 2010-03-30 22:31 溪流 閱讀(2384) 評論(15)  編輯 收藏 引用 所屬分類: C++

            評論:
            # re: 道德問題?論new操作失敗后的操作 2010-03-30 22:48 | OwnWaterloo
            最近很活躍嘛  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-03-30 22:54 | OwnWaterloo
            >>我需要給 vector(size_t size) 標記上 throw 嗎?如果不標記,使用者怎么知道這里可能會有異常?

            不需要, 不標記就是throw all。

            new失敗了
            1 :讓bad_alloc直接向上拋就是了
            為什么不需要大量的try catch?
            因為向上拋的過程中會析構棧上的對象, 回滾狀態, 并找到一個處理器。
            你會將代碼寫成異常安全的, 是吧?

            2: 采用兩段式。
            其實就是使用返回狀態代碼的處理方式了。

            在某個函數f中, 先構造一個半成品, 在使用之前create或者怎樣。
            如果失敗, 就通過狀態碼向f的調用者報告。
            f的調用者g又可能向g的調用者繼續報告。
            直到找到一個能處理的地方。

            這其實和異常是相同的, 只是異常對這些過程是自動的。
              回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-03-31 00:14 | 溪流
            @OwnWaterloo
            其實我不是很清楚什么叫異常安全。不知道異常安全是盡量避免寫出 try catch 還是盡量到處寫 try catch。但我潛意識里不喜歡寫 try catch,也不喜歡用會拋異常的東東,如 MFC 中的 CFile。  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作[未登錄] 2010-03-31 08:55 | chentan
            樓主最近寫的都是非常敏感的話題  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-03-31 09:02 | 欣萌
            Mark  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-03-31 09:14 | 溪流
            @chentan
            是嗎?哈哈~  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-03-31 09:15 | 溪流
            @OwnWaterloo
            因為最近又想起了那些困惑的事~~  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-03-31 10:39 | ljbxc
            我也常遇到,很煩人的問題。  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-03-31 14:52 | OwnWaterloo
            @溪流
            這樣很好。 不要悶頭只顧寫代碼; 花一些時間思考。  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-03-31 15:42 | 陳梓瀚(vczh)
            你只要覺得,你那一行發生的錯誤的話那么你的class就不能被創建,那就在那里拋異常。這樣可以使得你的class在那種情況下不可能被創建成功。  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-04-03 15:06 | 溪流
            @陳梓瀚(vczh)
            那我該期待別人怎么用我的class呢?  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-04-06 15:39 | only
            你可以去認真看看《effective C++ 》(第三版) 條款49-52!
            相信對于你會很有用,樓主!  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作 2010-04-06 15:54 | 溪流
            @only
            它是不是講如何利用什么handler去釋放內存之類的?
            我的題設是內存申請失敗,而不是被new的那個對象的構造函數執行失敗。(我已經假設了被new的只是一個int)
            我的困惑不是技術上如何保證沒有內存泄漏,而是——
            要是我的構造函數有異常拋出,用戶該如何用這個類?我該不該讓構造函數拋出異常?  回復  更多評論
              
            # re: 道德問題?論new操作失敗后的操作[未登錄] 2010-04-14 15:11 | siwei
            @溪流
            有意思的話題。Symbian中的方式是重載了new,以及應用兩段構造。  回復  更多評論
              
            久久天天日天天操综合伊人av| 国内精品伊人久久久久av一坑 | 国产精品99久久99久久久| 国色天香久久久久久久小说| 久久综合88熟人妻| 国产精品99久久99久久久| 国产女人aaa级久久久级| 久久精品国产日本波多野结衣| 久久综合给久久狠狠97色| 93精91精品国产综合久久香蕉| 欧美日韩精品久久久免费观看| 婷婷五月深深久久精品| 久久人人爽人人精品视频| 久久亚洲私人国产精品| 久久久久久噜噜精品免费直播 | 亚洲AV乱码久久精品蜜桃| 青青草原综合久久大伊人精品| 久久久久久午夜精品| 久久香蕉国产线看观看乱码| 国内精品久久久久影院亚洲| 国产农村妇女毛片精品久久| 久久一日本道色综合久久| 亚洲伊人久久综合中文成人网| 久久综合中文字幕| 久久永久免费人妻精品下载| 国内精品久久久久影院老司| 久久九九久精品国产| 精品久久香蕉国产线看观看亚洲| 久久无码专区国产精品发布| 久久久久香蕉视频| 香蕉久久一区二区不卡无毒影院| 久久久婷婷五月亚洲97号色| 国产一区二区久久久| 婷婷久久综合| 亚洲精品99久久久久中文字幕| 午夜不卡888久久| 国产精品va久久久久久久| 色噜噜狠狠先锋影音久久| 青青青青久久精品国产h| 久久精品国产亚洲综合色| 久久久精品2019免费观看|