???? c++在數據庫開發這種需要進行快速開發的地方一直是以效率低下著稱,borland 的c++ build 依賴vcl提供的組件勉強可以實現,但是由于用的是pascal的東西,根基不行,一直叫不響.再看看現在流行的數據庫,sql server 和mysql 等雖然提供本地開發的頭文件和庫,但是對于一些可能使用不同數據庫的應用來說,選擇特定某種數據庫開發顯然是不妥,因此,象ado這種數據操作接口大行其道.這又給用c++開發帶來了麻煩,象vb等工具可以自動轉換variant類型,但是在vc使用的時候卻很麻煩.網上那些使用ado接口的c++的例子,大多數都從#import 那個dll開始, 然后就是繁瑣的創建instance,打開記錄集,最后還要進行一個variant到c++類型的轉換,確實是夠麻煩的,難怪效率低下.
?
??? 在很多情況下,一些數據查詢和數據操作往往是在開發的時候就確定下來了,比如說你有個用戶表要操作,你在開發的時候就知道這個表中的所有字段名稱和字段類型,但是你用ado,你卻還是要和繁瑣的variant打交道,把它轉換為你想要的數據類型. 如果能實現一個類,在知曉數據表字段類型的情況下,可以方便地使用c++數據類型,并且可以進行諸如插入,刪除,更新等操作,那就方便多了.? 這就是本文想實現的目標.? 但是對于那些在運行時候才確定的數據操作,如用戶輸入一個 sql 查詢,然后再來進行操作,對于象c++ 這種強類型的語言來說,可能除了使用 variant別無他法, 對于這方面,本文沒有給出解決辦法.
本文中的代碼在vc6下編譯通過
需要的庫 loki for vc6, boost
一. 構造記錄結構
?? 通常來說,數據集由一條條的記錄組成,記錄下有字段,記錄是數據操作的基本單元.照c++的習慣,最好是一條記錄一個結構,字段就是這個結構的成員,然后用列表或者vector組合一下就是記錄集了.但是一條記錄的字段是有可能變動的,還好,有loki 的 typelist.
?
?1?#if?!defined(INCLUDE_40697E96_B6DE_449D_A516_B0376650A488)
?2?#define?INCLUDE_40697E96_B6DE_449D_A516_B0376650A488
?3?
?4?#if?_MSC_VER?>?1000
?5?#pragma?once
?6?#endif
?7?
?8?#include?<typelist.h>
?9?#include?<functor.h>
10?#include?<boost/preprocessor/dec.hpp>
11?
12?#define?HELP_2(a?,?b)?a(b)
13?
14?#define?MEM_STRUCT_MAKE_IMPL(n)?template?<typename?_tlist>?\
15?struct?In?:?public?_str_impl_help<n>::template?In<typename?_tlist::Tail>?\
16?{?\
17?????typedef?_str_impl_help<n>::template?In<typename?_tlist::Tail>?_Base;?\
18?????enum?{????en_member_count?=?_Base::en_member_count+1,?};?\
19?????typedef?typename?::Loki::TL::TypeAt<_tlist,?0>::Result?_type##n;?\
20?????_type##n?m_P##n;?\
21?????__declspec(property(get?=?GetF##n,?put?=?PutF##n))?\
22?????_type##n?Field##n;?\
23?????_type##n?GetF##n()?{?return?m_P##n;?}?\
24?????void?PutF##n(const?_type##n?val)?{?m_P##n?=?val;?}?\
25?};?\
26?}
27?
28?#define?MEM_STRUCT_MAKE(n)?template?<>?\
29?????struct?_str_impl_help<n>?{?\
30?????HELP_2(MEM_STRUCT_MAKE_IMPL,?BOOST_PP_DEC(n))
31?????
32?
33?namespace?impl_help
34?{
35?????template?<int?_p>
36?????struct?_str_impl_help;
37?
38?????template?<>
39?????struct?_str_impl_help<0>?{};
40?????template?<>
41?????struct?_str_impl_help<1>?{
42?????????template?<typename?_tlist>?
43?????????struct?In?{
44?????????????enum?{
45?????????????????en_member_count?=?1,
46?????????????};
47?
48?????????????typename?::Loki::TL::TypeAt<_tlist,?0>::Result?m_P0;
49?
50?????????????__declspec(property(get=GetF0,?put=PutF0))
51?????????????typename?::Loki::TL::TypeAt<_tlist,?0>::Result?Field0;
52?
53?????????????typename?::Loki::TL::TypeAt<_tlist,?0>::Result?GetF0()?{?
54?????????????????return?m_P0;?
55?????????????}
56?
57?????????????void?PutF0(const?typename?::Loki::TL::TypeAt<_tlist,?0>::Result?val)?{
58?????????????????m_P0?=?val;
59?????????????}
60?????????};
61?????};
62?
63?????MEM_STRUCT_MAKE(2);
64?????MEM_STRUCT_MAKE(3);
65?????MEM_STRUCT_MAKE(4);
66?????MEM_STRUCT_MAKE(5);
67?????MEM_STRUCT_MAKE(6);
68?????MEM_STRUCT_MAKE(7);
69?????MEM_STRUCT_MAKE(8);
70?????MEM_STRUCT_MAKE(9);
71?????MEM_STRUCT_MAKE(10);
72?????MEM_STRUCT_MAKE(11);
73?????MEM_STRUCT_MAKE(12);
74?????MEM_STRUCT_MAKE(13);
75?????MEM_STRUCT_MAKE(14);
76?????MEM_STRUCT_MAKE(15);
77?}
78?
79?template?<typename?_tlist>
80?struct?struct_mem?:?public?impl_help::_str_impl_help<?::Loki::TL::Length<_tlist>::value?>::?
81?????template?In<?::Loki::TL::Reverse<_tlist>::Result?>
82?{
83?};
84?
85?
86?#endif?//!defined(INCLUDE_40697E96_B6DE_449d_A516_B0376650A488)
上面是模板實現根據指定typelist,生成擁有相應成員的結構.
typedef struct_mem< TYPELIST_6(DB_STRING, DB_STRING, DB_STRING, DB_STRING, DB_STRING, DB_STRING) > CustomerRow; 就可以聲明一個有6個成員的結構了.
上面的代碼使用template的特化到有15個成員的結構,全部寫出來的話,重復而且麻煩,所以使用了宏來幫助生成.
?