【VC經驗】偶的小經驗一


說明:這些是我在編程中碰到和處理過的一些問題和經驗。這是第一篇(看看效果,先)。

一,PostMessage和SendMessage

1, PostMessage只把消息放入隊列,不管其他程序是否處理都返回,然後繼續執行,這是個異步消息投放函數。而SendMessage必須等待其他程序處理消息完了之後才返回,繼續執行,這是個同步消息投放函數。而且,PostMessage的返回值表示PostMessage函數執行是否正確;而SendMessage的返回值表示其他程序處理消息後的返回值。這點大家應該都明白。

2, 如果在同一個線程內,PostMessage發送消息時,消息要先放入線程的消息隊列,然後通過消息循環Dispatch到目標窗口。SendMessage發送消息時,系統直接調用目標窗口的消息處理程序,並將結果返回。SendMessage在同一線程中發送消息並不入線程消息隊列。

如果在不同線程內。最好用PostThreadMessage代替PostMessage,他工作的很好。SendMessage發送消息到目標窗口所屬的線程的消息隊列,然後發送消息的線程等待(事實上,他應該還在做一些監測工作,比如監視QS_SENDMESSAGE標誌),直到目標窗口處理完並且結果返回,發送消息的線程才繼續運行。這是SendMessage的一般情況,事實上,處理過程要複雜的多。比如,當發送消息的線程監測到有別的窗口SendMessage一個消息到來時,他直接調用窗口處理過程(重入),並將處理結果返回(這個過程不需要消息循環中GetMessage等的支持)。

3, msdn: If you send a message in the range below WM_USER to the asynchronous message functions (PostMessage, SendNotifyMessage, and SendMessageCallback), its message parameters can not include pointers. Otherwise, the operation will fail.

如果發送的消息碼在WM_USER之下(非自定義消息)且消息參數中帶有指針,那麼PostMessage,SendNotifyMessage,SendMessageCallback這些異步消息發送函數將會調用失敗。

最好不要用PostMessage發送帶有指針參數的消息。

二,設置視圖(客戶區)大小

在主窗口中(也可以改到視圖中,用GetParentFrame獲得主窗口指針):

CRect rect(100,100,400,400); //要設置的視圖的位置和大小

//AdjustWindowRectEx通過視圖大小計算出主窗口大小。
//函數執行時,rect傳入的是新視圖的位置和大小。
//第三個參數爲FALSE表示不需要計算菜單的尺寸,否則加入菜單尺寸計算窗口大小。
//函數執行完之後,rect中返回的是新的窗口位置和大小。

::AdjustWindowRectEx(&rect, GetStyle(), FALSE, GetExStyle()); 

MoveWindow(&rect);   //移動窗口

註釋:用這種方法用來動態改變視圖大小。要注意的問題是,AdjustWindowRectEx第三個參數爲TRUE的話,加入計算的菜單是一行的高度,如果有多行(可以通過客戶區尺寸是不是高度沒有達到要求),需要用:
int dh = GetSystemMetrics(SM_CYMENU);
得到菜單高度,然後添加到窗口高度中。

三,ADO數據庫操作

1, 得到數據庫中的表名,某個表的字段名。用 rs = m_pConnection ->OpenSchema (SchemaEnum QueryType,_variant_t param,_variant_t param2);可以得到。

比如
rs = m_curConnection.OpenSchema(adSchemaColumns,vtMissing,vtMissing);
得到的一個記錄集。他的字段包括: TABLE_SCHEMA,TABLE_NAME,COLUMN_NAME等很多信息。效果圖如下(部分):

效果圖


可以用這個rs來移動記錄,獲得表名(字段"TABLE_NAME"下),某個表的字段名(字段"COLUMN_NAME")。需要注意的是,得到表名的時候要判斷得到的表名是否重複(和上次得到的表名比較一下就可以拉);在表明相同的時候,他的字段名("COLUMN_NAME"下)是不同的,這樣可以得到某個表的所有字段名。

2, 用好recordset對象的Supports(CursorOptionEnum CursorOptions)方法。有時候用來判斷數據庫是否支持某類操作。比如:

//如果可讀取並設置 AbsolutePosition 和 AbsolutePage 的屬性
if(m_curRs.Supports(adApproxPosition)) {
 //...
}

關於CursorOptionEnum的取值,很多ado手冊上都有。

 

enoloo,

6.4,2004

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章