???? c++在數(shù)據(jù)庫(kù)開發(fā)這種需要進(jìn)行快速開發(fā)的地方一直是以效率低下著稱,borland 的c++ build 依賴vcl提供的組件勉強(qiáng)可以實(shí)現(xiàn),但是由于用的是pascal的東西,根基不行,一直叫不響.再看看現(xiàn)在流行的數(shù)據(jù)庫(kù),sql server 和mysql 等雖然提供本地開發(fā)的頭文件和庫(kù),但是對(duì)于一些可能使用不同數(shù)據(jù)庫(kù)的應(yīng)用來說,選擇特定某種數(shù)據(jù)庫(kù)開發(fā)顯然是不妥,因此,象ado這種數(shù)據(jù)操作接口大行其道.這又給用c++開發(fā)帶來了麻煩,象vb等工具可以自動(dòng)轉(zhuǎn)換variant類型,但是在vc使用的時(shí)候卻很麻煩.網(wǎng)上那些使用ado接口的c++的例子,大多數(shù)都從#import 那個(gè)dll開始, 然后就是繁瑣的創(chuàng)建instance,打開記錄集,最后還要進(jìn)行一個(gè)variant到c++類型的轉(zhuǎn)換,確實(shí)是夠麻煩的,難怪效率低下.
?
??? 在很多情況下,一些數(shù)據(jù)查詢和數(shù)據(jù)操作往往是在開發(fā)的時(shí)候就確定下來了,比如說你有個(gè)用戶表要操作,你在開發(fā)的時(shí)候就知道這個(gè)表中的所有字段名稱和字段類型,但是你用ado,你卻還是要和繁瑣的variant打交道,把它轉(zhuǎn)換為你想要的數(shù)據(jù)類型. 如果能實(shí)現(xiàn)一個(gè)類,在知曉數(shù)據(jù)表字段類型的情況下,可以方便地使用c++數(shù)據(jù)類型,并且可以進(jìn)行諸如插入,刪除,更新等操作,那就方便多了.? 這就是本文想實(shí)現(xiàn)的目標(biāo).? 但是對(duì)于那些在運(yùn)行時(shí)候才確定的數(shù)據(jù)操作,如用戶輸入一個(gè) sql 查詢,然后再來進(jìn)行操作,對(duì)于象c++ 這種強(qiáng)類型的語言來說,可能除了使用 variant別無他法, 對(duì)于這方面,本文沒有給出解決辦法.
本文中的代碼在vc6下編譯通過
需要的庫(kù) loki for vc6, boost
一. 構(gòu)造記錄結(jié)構(gòu)
?? 通常來說,數(shù)據(jù)集由一條條的記錄組成,記錄下有字段,記錄是數(shù)據(jù)操作的基本單元.照c++的習(xí)慣,最好是一條記錄一個(gè)結(jié)構(gòu),字段就是這個(gè)結(jié)構(gòu)的成員,然后用列表或者vector組合一下就是記錄集了.但是一條記錄的字段是有可能變動(dòng)的,還好,有l(wèi)oki 的 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)
上面是模板實(shí)現(xiàn)根據(jù)指定typelist,生成擁有相應(yīng)成員的結(jié)構(gòu).
typedef struct_mem< TYPELIST_6(DB_STRING, DB_STRING, DB_STRING, DB_STRING, DB_STRING, DB_STRING) > CustomerRow; 就可以聲明一個(gè)有6個(gè)成員的結(jié)構(gòu)了.
上面的代碼使用template的特化到有15個(gè)成員的結(jié)構(gòu),全部寫出來的話,重復(fù)而且麻煩,所以使用了宏來幫助生成.
?