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

woaidongmao

文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
數(shù)據(jù)加載中……

淺析C++ Compile-time Assertion技術

你可能經(jīng)常需要利用運行時斷言技術,它可以方便地測試前提條件。但是,隨著Metaprogramming概念的出現(xiàn),編譯時斷言技術也已經(jīng)和runtime assertion一樣的普遍了。如何在編譯時進行斷言呢?其實,方法只有一個,就是讓編譯器生成一條錯誤信息,但是編譯器生成的錯誤信息信息性往往有又理想。并且,即使你在一種編譯上設計了一種方案,你也很難把它移植到其他的編譯器上。我們通過其實現(xiàn)方法的改進和一個Boost中的例子,來看看如何更好的實現(xiàn)這種技術。

例如,你需要一個安全的類型轉換機制,它只允許你把個頭小的類型轉換為個頭大的類型。此時,就可以利用Compile-time Assertion解決這個問題。

template <typename To, typename From>

To safe_reinterpret_cast(From from) {

    assert(sizeof(To) >= sizeof(From));

    return reinterrupt_cast (from);

};

而后,就像你使用同樣的 C++ 類型轉換一樣來使用這個 safe_reinterpret_cast

long l = 255;

short s = safe_reinterpret_cast<short>(l);

這樣一來,你就可以確保只有在小 à 大的轉換才是正確的,如果進行非法的轉換,就會在運行時發(fā)生斷言。

顯然,如果能夠在編譯時給用戶指出代碼中的問題更為合適一些。如果這個轉換只在程序很少被執(zhí)行到的一個分支上被執(zhí)行,那么當你把它移植到一個新的編譯器上或平臺上的時候,你就有可能忘記程序中所有不可移植的部分,例如上面提到的 reinterrupt_cast ,從而給你的程序帶來不必要的 bug

其實,上面我們被評估的表達式是一個編譯器常量,也就是說你完全有可以讓編譯器取代運行時代碼來進行檢查。解決的思路是在表達式為 true 的時候給編譯器傳遞正確的代碼,而在表達式為 false 的時候給編譯器提供一個語法錯誤的代碼,這樣,當被評估的表達式為 0 的時候,編譯器就會發(fā)出一個錯誤信號。

最簡單的 compile-time assertion 解決方案是 Van Horn 1997 年提出的,它可以在 C C++ 的代碼中工作,依賴的條件很簡單,數(shù)組的長度不能為 0

#define STATIC_CHECK(expr) { char unnamed[(expr ? 1 : 0)]; }

現(xiàn)在,如果你寫下下面的代碼:

template <typename To, typename From>

To safe_reinterpret_cast(From from) {

    STATIC_CHECK(sizeof(To) >= sizeof(From));

    return reinterpret_cast (from);

};

… …

void * somePointer = 0;

char c = safe_reinterpret_cast<char>(somePointer);

如果 void* 的長度小于 char( 這個并沒有在目前的 C++ 標準的規(guī)定 ) ,編譯器就會告訴你創(chuàng)建了一個長度為 0 的數(shù)組。

問題是這個方法提供的錯誤信息并不是很說明問題。“不能創(chuàng)建長度為0的數(shù)組”并不能表示“char類型放不下一個指針”。這種方法很難想用戶提供customized message。錯誤信息的來源并不是因為代碼違法了程序設計的意圖,而是因為破壞了某些語法規(guī)則。

更好的解決方案是依賴一個模板提供一個具有說明性的名字,這樣,編譯器就會在錯誤信息中包含這個名字了。

template <bool> struct CompileTimeError;

template <> struct CompileTimeError<true> {};

#define STATIC_CHECK1(expr1) { (CompileTimeError<(expr1) != 0>()); }

CompileTimeError 帶有一個非類型參數(shù),并且只有 true 的特化版本,這樣,當被評估的表達式不滿足條件時,編譯器就會抱怨沒有 CompileTimeError 的特化版本,這個比剛才的錯誤多多少少要好一些。

當然,這個設計仍然有很大的擴展空間。因為我們還是沒有辦法來訂制錯誤消息。一個簡單的辦法就是在 STATIC_CHECK 中加入一個消息參數(shù),然后讓這個消息參數(shù)在錯誤信息中顯示。這個方法也有自己的缺點,就是你必須要保證傳遞給 C++ 的這個錯誤消息參數(shù)一定是合法的。于是我們可以對于上面的 CompileTimeError 做以下的改進:

template <bool> struct CompileTimeChecker {

    CompileTimeChecker(...) {};

};

template <> struct CompileTimeChecker<false> { };

 

#define STATIC_CHECK2(expr2, msg) {\

    class ERROR_##msg {}; \

    sizeof((CompileTimeChecker<(expr2!=0)>((ERROR_##msg()))));\

}

template <typename To, typename From>

To safe_reinterpret_cast(From from) {

    STATIC_CHECK2((sizeof(To) >= sizeof(From)),

Destination_Type_To_Narrow);

    return reinterpret_cast (from);

};

這樣,當你仍舊使用剛才的代碼時:

void * somePointer = 0;

char c = safe_reinterpret_cast<char>(somePointer);

由于 CompileTimeChecker 可以接受任意參數(shù),而特化的 CompileTimeChecker 并沒有這樣的構造函數(shù),這樣,當被評估的表達式為 0 的時候,就會出現(xiàn)編譯時錯誤

cannot convert

from

'safe_reinterpret_cast::ERROR_Destination_Type_To_Narrow'

to

'CompileTimeChecker'

這次的錯誤信息變的比較有提示性了。

現(xiàn)實中的應用——BOOST_STATIC_ASSERT & boost::checked_delete

BOOST_STATIC_ASSERT

boost/static_assert.hpp中定義了一個宏BOOST_STATIC_ASSERT,用于完成編譯時靜態(tài)檢查。其實現(xiàn)方式了我們的第2種方式很類似,利用了模板的特化技術

#define BOOST_STATIC_ASSERT( B ) \

   typedef ::boost::static_assert_test<\

      sizeof(::boost::STATIC_ASSERTION_FAILURE< (bool)( B ) >)>\

         BOOST_JOIN(boost_static_assert_typedef_, __COUNTER__)

其中:

template <int x> struct static_assert_test{};

#define BOOST_JOIN( X, Y ) X##Y

template <bool x> struct STATIC_ASSERTION_FAILURE;

template <> struct STATIC_ASSERTION_FAILURE<true> { enum { value = 1 }; };

這里,只為 true 類型進行了特化,這樣,當我們嘗試聲明一個 STATIC_ASSERTION_FAILURE<false> 的時候就會引發(fā)編譯時錯誤。

這樣,整個宏的含義就是做了一個 typedef:

typedef ::boost::static_assert_test<evaluate condition> boost_static_assert_typedef___COUNTER__

而只有當evaluate conditiontrue的時候,這樣的typedef才是正確的,從而實現(xiàn)了編譯時斷言(上面的代碼只是msvc的實現(xiàn),對不同的編譯器實現(xiàn)略有不同,但是思想是類似的)。

例子:確保一個模板參數(shù)的類型只能是整數(shù)

template <typename T> class only_compatible_with_integral_types {

    BOOST_STATIC_ASSERT(boost::is_integral::value);

};

之后,如果你使用下面的定義:

only_compatible_with_integral_types<double> test2;

就會引發(fā)編譯錯誤:

use of undefined type 'boost::STATIC_ASSERTION_FAILURE

boost::checked_delete

當我們利用指針刪除一個對象的時候,對象類型是否完整決定了對象是否能夠被正確刪除。但是,如果你用 delete 去刪除一個類型并不完整的對象的指針,編譯器并不會給你提供任何錯誤信息,但是這樣做的結果卻是對象的析構函數(shù)根本就沒有被調(diào)用。

checked-delete 定義在 boost/checkd_delete.hpp 中,它可以保證在你摧毀一個對象的時候,必須對該對象的類型有完全的了解。先來看個例子:

#include

class some_class;

some_class* create() {

  return (some_class*)0;

}

int main() {

  some_class* p=create();

  boost::checked_delete(p2);

}

編譯器就會抱怨 some_calss 是一個不完整的類型。在我們進一步去了解解決方案之前,我們先來看一個由于不完整類型帶來的 memory leak 的例子:

// in deleter.h

class to_be_deleted;

class deleter {

public :

    void delete_it(to_be_deleted* p);

};

// in deleter.cpp

#include "deleter.h"

 

void deleter::delete_it(to_be_deleted* p) {

    delete p; // !!!memory leak here

}

// in to_be_deleted.h

#include

 

class to_be_deleted {

    class test {

    public:

       test() {};

       ~test() { std::cout<<"I'm destructed correctly!"<

    };

    test* p;

public :

    to_be_deleted() { p = new test(); };

    ~to_be_deleted() {

       delete p;

       std::cout<<"I've important things to say!"<

    }

};

之后用下面的測試代碼:

#include "deleter.h"

#include "to_be_deleted.h"

int main() {

    to_be_deleted* p = new to_be_deleted();

    deleter d;

    d.delete_it(p);

    return 0;

}

你會發(fā)現(xiàn), to_be_deleted 的析構函數(shù)并沒有被調(diào)用,原因在于 deleter.cpp 中,并沒有包含 to_be_deleted.h ,這樣, delete 對于齊要刪除的指針一無所知,導致了析構函數(shù)并沒有真正被調(diào)用。

解決的方法也很簡單,利用 boost::checked_delete 進行刪除。

#include

#include "deleter.h"

void deleter::delete_it(to_be_deleted* p) {

    //delete p; // memory leak here

    boost::checked_delete(p);

}

這時,編譯器便會抱怨說 to_be_deleted 是未知的類型。其實 ,checked_delete 的實現(xiàn)原理是非常簡單的,只是說對于未知類型,使用 sizeof 運算符會返回 0 ,而 C++ 并不允許創(chuàng)建長度為 0 的數(shù)組。如下所示:

template <class T> inlinevoid checked_delete(T * x)

{

    // intentionally complex - simplification causes regressions

    typedef char type_must_be_complete[ sizeof(T)? 1: -1 ];

    (void) sizeof(type_must_be_complete);

    delete x;

}

posted on 2008-09-15 13:39 肥仔 閱讀(644) 評論(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>
            国产日韩欧美视频在线| 一区二区黄色| 亚洲国产片色| 久久裸体视频| 免费视频久久| 亚洲电影观看| 亚洲最新色图| 午夜日韩在线观看| 欧美在线一级va免费观看| 久久中文字幕导航| 欧美日韩亚洲三区| 国产美女高潮久久白浆| 亚洲国产成人在线播放| 亚洲欧洲一区二区三区在线观看| 亚洲免费高清视频| 欧美中文在线免费| 亚洲第一成人在线| 亚洲一区二区三区涩| 久久精品青青大伊人av| 欧美高清在线精品一区| 国产精自产拍久久久久久蜜| 在线播放不卡| 亚洲永久免费| 欧美1区2区3区| 亚洲制服欧美中文字幕中文字幕| 久久精品国产视频| 欧美午夜电影在线| 亚洲高清在线视频| 欧美亚洲在线| 亚洲国产成人久久综合一区| 性欧美videos另类喷潮| 免费精品视频| 激情文学一区| 欧美在线看片| 亚洲美洲欧洲综合国产一区| 久久国产精品一区二区三区四区| 欧美日本乱大交xxxxx| 国产永久精品大片wwwapp| 亚洲天堂av在线免费| 麻豆精品一区二区av白丝在线| 亚洲人成在线观看一区二区| 久久精精品视频| 国产精品丝袜91| 亚洲一区二区免费看| 欧美激情一二三区| 欧美一区二区在线看| 国产精品国产三级国产aⅴ浪潮 | 亚洲二区在线| 久久久国产一区二区| 久久成人18免费网站| 亚洲综合精品| 国产精品入口尤物| 久久美女性网| 欧美成人一区二免费视频软件| 雨宫琴音一区二区在线| 欧美大片免费久久精品三p | 老司机凹凸av亚洲导航| 尤物网精品视频| 伊人久久亚洲热| 久久精品30| 午夜精品区一区二区三| 国产精品视频1区| 欧美一区二区三区四区夜夜大片| 亚洲人成毛片在线播放| 欧美国产日产韩国视频| 亚洲美女精品久久| 亚洲美女av在线播放| 欧美日韩一区二| 亚洲男人的天堂在线观看| 日韩视频精品| 国产精品任我爽爆在线播放| 午夜欧美不卡精品aaaaa| 亚洲女同性videos| 国产一区二区视频在线观看 | 久久久精品免费视频| 欧美成人精品在线| 日韩视频中文字幕| 欧美视频一区二区三区四区 | 亚洲乱码日产精品bd| 亚洲国内在线| 欧美人与性动交a欧美精品| 一区二区日韩免费看| 亚洲一区欧美一区| 韩国一区二区三区在线观看| 噜噜噜在线观看免费视频日韩| 久久久久久久久久看片| 99热这里只有精品8| 亚洲视频免费| 黑人操亚洲美女惩罚| 最新国产の精品合集bt伙计| 欧美日韩视频在线| 久久精品女人| 欧美日韩成人综合在线一区二区| 香蕉成人久久| 亚洲二区在线视频| 午夜精品福利电影| 亚洲高清在线观看一区| 欧美日韩mp4| 久久久久99| 欧美精品自拍| 久久婷婷影院| 欧美女人交a| 久久久久久伊人| 欧美成年网站| 久久aⅴ国产紧身牛仔裤| 欧美黑人在线播放| 久久免费的精品国产v∧| 欧美午夜a级限制福利片| 久久人人97超碰国产公开结果 | 亚洲视频二区| 亚洲欧美日韩成人| 亚洲精品一区二区三| 欧美一级久久久| 99国产精品一区| 久久久久久综合| 欧美一级久久久| 欧美日韩视频免费播放| 亚洲高清123| 在线免费不卡视频| 午夜在线不卡| 午夜久久资源| 欧美日韩国产探花| 欧美国产日韩免费| 狠狠爱成人网| 性xx色xx综合久久久xx| 午夜精品一区二区三区在线播放| 欧美精品一区二区三区很污很色的| 久久久久久网址| 国产精品www994| 一本色道久久88综合日韩精品| 日韩视频在线观看| 久久亚洲一区二区| 免费日韩成人| 在线成人免费观看| 久久国产精品99国产| 久久久成人精品| 好吊色欧美一区二区三区视频| 亚洲在线国产日韩欧美| 亚洲欧美日韩在线高清直播| 欧美日韩国产经典色站一区二区三区| 亚洲福利视频专区| 亚洲精品乱码久久久久久按摩观 | 国产精品露脸自拍| 午夜精品免费视频| 亚洲一线二线三线久久久| 欧美午夜a级限制福利片| 亚洲美女视频在线免费观看| 亚洲人成在线观看网站高清| 母乳一区在线观看| 亚洲人妖在线| 欧美激情在线有限公司| 91久久夜色精品国产九色| 99视频热这里只有精品免费| 欧美日韩一区二区三区在线| 一区二区精品在线观看| 欧美+亚洲+精品+三区| 一区二区高清视频| 国产一级精品aaaaa看| 欧美在线啊v| 你懂的国产精品永久在线| 欧美粗暴jizz性欧美20| 久久麻豆一区二区| 羞羞答答国产精品www一本| 夜夜精品视频一区二区| 亚洲精华国产欧美| 欧美电影免费| 免费成人小视频| 久久综合影音| 久久久久久高潮国产精品视| 久久国产精品久久久久久| 亚洲先锋成人| 一本久久综合亚洲鲁鲁五月天| 亚洲国产成人久久综合| 国产在线精品成人一区二区三区 | 免费永久网站黄欧美| 久久久久久久久岛国免费| 欧美亚洲一区在线| 性色一区二区三区| 欧美在线播放一区| 新67194成人永久网站| 午夜影视日本亚洲欧洲精品| 午夜久久一区| 欧美自拍偷拍| 久久综合久久美利坚合众国| 久久亚洲综合| 欧美 日韩 国产一区二区在线视频| 久久综合亚洲社区| 欧美韩日一区二区三区| 欧美日本一区| 国产精品美女在线观看| 国产三级欧美三级日产三级99| 国产女优一区| 一区二区三区在线免费播放| 在线播放亚洲一区| 亚洲精品久久久久久下一站| 一区二区日韩免费看| 亚洲综合视频在线| 久久久久久精| 亚洲国产精品传媒在线观看| 亚洲最新在线视频|