早就聽說bcb(borland c++ builder)是一個強大的RAD開發工具,也早就聽說曾經的borland搞出的編譯器堪稱經典。
恰好最近在做一個GUI工具,想在界面開發上盡量快一點。每一次用上MFC都讓我覺得渾身難受,總有些常用的
界面功能它就是沒有。在接口實現上,MFC基本上就只是封裝了WIN API而已。想想世界上還有什么強大的GUI庫,
找了一下,其實不管GUI庫封裝的怎么樣,我更多地還是需要一個工具,能夠快速地堆積出界面。
于是,在網上下載了被國人精簡了的bcb2009。然后,噩夢開始了。首先,我需要把邏輯層代碼(也就是實現具體
功能的那一層)移植到BCB下。然后得到了很多和語法相關的編譯錯誤:
1.
E2397: Template argument cannot have static or local linkage
這個錯誤發生于:
void func()
{
struct Info
{
};
std::queue<Info> abc;
}
它的意思是,模板參數必須是全局鏈接的,總之它不允許std::queue的參數是一個在函數內部臨時定義
的類型(誰來告訴我這是C++標準)。
2.
E2357 Reference initialized with 'FileLoader::RawData', needs lvalue of type 'FileLoader::RawData'
這個錯誤發生于:
FileLoader::RawData FileLoader::GetRawData() const;
FileLoader::RawData &raw = loader.GetRawData(); //不能用引用
很久沒看C++書,所以,誰又來告訴我C++標準里,這里到底能不能用引用?
3.
E2515 Cannot explicitly specialize a member of a generic template class
這個錯誤發生的情景更復雜些:
template <typename _Tp>
class Test
{
template <typename _U>
class Other;
template <>
class Other<void>
{
};
};
意思是說,我不能在一個模板類里特化成員模板類。誰又來告訴我標準規定的是什么?
4.
void func( Obj &a )
{
}
func( Obj() );這個也被視為錯誤。必須得在調用func之前自己定義個臨時變量。
5.
我曾經留下了關于宏遞歸的一些代碼,被用在我寫的lua-binder和lua-caller中自動生成代碼。這下好了,
BCB開始警告我,我的這些宏不能工作了。它和MSVC在某些事情上分歧可真是大:
#define PARAM( n ) ,typename P##n //注意這個宏包含一個逗號
#define CHR( x, y ) CHR1( x, y )
#define CHR1( x, y ) x##y
#define BCB_ERROR( a, b ) CHR( a, b )
BCB_ERROR( 1, PARAM( 1 ) ) 當這樣使用宏時,基于我在GNU C上看到的關于宏的規則,會先展開
PARAM(1),于是得到BCB_ERROR( 1, ,typename P2 )。然后,BCB認為PARAM(1)展開的逗號需要參與
BCB_ERROR的展開了。于是,我的整個宏庫無法工作了。
關于這個問題,我直接用MSVC寫了個生成器,讓MSVC替我生成各種參數的lua-binder和lua-caller,然后
寫成外部頭文件,最后直接在BCB里包含了這些頭文件。從而使我的lua-binder和lua-caller可以繼續使用。
然后,我的1W多行代碼終于在BCB下50多個WARNINGS的提示下編譯成功了。懷揣著興奮的心情,想自己終
于可以rapid開發界面了。創建了個VCL FORM APPLICATION,噩夢又開始了:
1.
BCB莫名其妙地在我編譯一個CPP文件時給出如下提示:
F1004 Internal compiler error at 0x59b4ea8 with base 0x5980000
看起來像是BCB的編譯器給崩潰了。囧。google了一下,發現不是我人品問題,很多人遇到相同的問題。
別人給出的解決方案是:restart your bcb。從昨天晚上到現在為止,這個錯誤發生了好幾次。
2.
new std::ofstream();會讓程序崩潰,往不該寫的地方寫了東西。我就奇怪了,你BCB自己帶的C++IO實現,
難道還有BUG?再次google,還真發現是BCB自己的BUG,并且在幾個版本之前就存在這個BUG。那個天真
的老外還說希望在BCB2009下能被修復。修改方案如下:
1)xlocale文件里把這句話注釋了:*(size_t *)&table_size = 1 << CHAR_BIT;
2)xlocale里把成員_Id_cnt訪問屬性改為public,然后在自己的文件里定義一次。
3.
程序終于可以運行了。但是BCB的IDE環境總是不那么貼心。我移動了幾個窗口改成我習慣的樣子,但是一重啟
居然又恢復成default(難道是因為盜版)。它的智能提示似乎總是跟著鼠標指針,有時候指向某個符號,鼠標
就顯示忙。為了提示某個類的成員,某個函數的原型,BCB偶爾都會卡一下。其實我不介意我的編輯器沒有這
些提示功能,在MSVC下我也從不用VA來幫我寫代碼。我甚至不厭其煩地在VIM下敲代碼切窗口去看函數原型,
但是,你他媽作為一個IDE就得像個IDE的樣子,要不,你干脆關掉所有功能,別給我卡就行了。
這個時候我開始懷疑選擇BCB會不會是一個錯誤的開始,或者說在使用某個東西時,總會帶著使用其他同類東西
的感覺甚至偏見去看待這個新事物。但是,在我想堅持繼續使用BCB時,我一compile,它又提示我:
F1004 Internal compiler error at 0x59b4ea8 with base 0x5980000