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

hdqqq

  C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
  35 隨筆 :: 0 文章 :: 104 評論 :: 0 Trackbacks


在前面的幾篇文章中,分別對ado操作接口,單條的記錄結(jié)構(gòu),variant數(shù)據(jù)轉(zhuǎn)換為c數(shù)據(jù)類型等做了說明,這些都是為最終實現(xiàn)一個方便的數(shù)據(jù)庫操作接口作準(zhǔn)備,在這一篇里,將創(chuàng)建一個模板數(shù)據(jù)庫記錄集操作類,它需要實現(xiàn)的功能有, select, update, 和delete操作,當(dāng)然,除了select sql是將在程序中給出外,其它的象update,delete等都應(yīng)該是這個模板記錄集根據(jù)字段類型和名稱自動生成的,而不必再人工生成這些sql語句. 用過微軟的.net的dataset應(yīng)該有類似的體驗,在vs2003中,通過wizzard生成和數(shù)據(jù)庫相關(guān)的dataset記錄集和datarow等,用戶只要在dataset中添加和刪除,最后update一下,dataset自動幫你生成sql并執(zhí)行.這樣就方便了很多.同時,記錄集的某條記錄的成員都是強(qiáng)數(shù)據(jù)類型的,而不是可變的variant數(shù)據(jù)類型. 對于經(jīng)常使用的跨表的數(shù)據(jù)查詢等,也可以使用,無需先定義數(shù)據(jù)結(jié)構(gòu)的麻煩.


template? < typename ?_tlist >
class?data_op_recordset
{
public :

????typedef?data_op_record_row
< _tlist > ?_DataRow;
????typedef?std::list
< ?_DataRow * ? > ?_DataSet;
????typedef?std::vector
< ?__tagFieldInfo? > ?_FieldNames;
????enum?{
????????en_field_count?
= ?_DataRow::en_member_count,
????};

public :
????virtual?BOOL?Load(
const ?TCHAR * ?query,?CCom_Connection & ?conn)?{
????????CCom_Recordset?lreset(__uuidof(Recordset));
????????_bstr_t?sql?
= ?query;
????????lreset
-> Open(sql,?(IDispatch * )conn,?ADODB_L::adOpenStatic,?ADODB_L::adLockReadOnly,?ADODB_L::adCmdText);
????????
if ?(m_DataSet.size())?{
????????????Clear();
????????}

????????GetFieldName(lreset);

????????
if ?(!lreset -> ADO_EOF? && ?!lreset -> BOF)?{
????????????
????????????lreset
-> MoveFirst();
????????????
int ?i;
????????????
for ?(i? = ? 0 ;?i? < ?lreset -> GetRecordCount();?i ++ )?{
????????????????_DataRow
* ?prow? = ? new ?_DataRow;
????????????????_data_get
< en_field_count > ::DoDataChange(prow,?m_FieldNames,?lreset);
????????????????m_DataSet.push_back(prow);
????????????????lreset
-> MoveNext();
????????????}
????????}
????????return?
0 ;
????}

????virtual?BOOL?Update(CCom_Connection
& ?conn)?{

????????BeforeUpdate();

????????BOOL?ret?
= ? FALSE ;
????????std::list
< ?CString? > ?sql_list;
????????
for (_DataSet::iterator?it? = ?m_DataSet.begin();?it?! = ?m_DataSet.end();?it ++ )?{
????????????CString?sql;
????????????
if ?(( * it) -> m_en_State? == ?_DataRow::_en_Row_Add)?{
????????????????sql?
= ?MakeInsertSql( * ( * it));
????????????}

????????????
if ?(( * it) -> m_en_State? == ?_DataRow::_en_Row_Update)?{
????????????????sql?
= ?MakeUpdateSql( * ( * it));
????????????}

????????????
if ?(( * it) -> m_en_State? == ?_DataRow::_en_Row_Del)?{
????????????????sql?
= ?MakeDeleteSql( * ( * it));
????????????}
????????????
if ?(sql.GetLength())?{

????????????????sql_list.push_back(sql);
????????????????TRACE(
" %s\n " ,?sql);
????????????}
????????}

????????
if ?(sql_list.size())?{
????????????conn
-> BeginTrans();
????????????try?{
????????????????std::list
< CString > ::iterator?it;
????????????????
for ?(it? = ?sql_list.begin();?it?! = ?sql_list.end();?it ++ )?{
????????????????????CString?cc;
????????????????????_bstr_t?temp;
????????????????????
????????????????????_variant_t?RecordsAffected;
????????????????????cc?
= ? * it;
????????????????????cc.Replace(
' \'','\"');
????????????????????TRACE( " %s\n " ,?cc);
????????????????????temp?
= ?cc;
????????????????????conn
-> Execute (temp,? & RecordsAffected,?ADODB_L::adCmdText);
????????????????}
????????????}catch?(_com_error?
& e)?{
????????????????CString?info;
????????????????_bstr_t?es,?ds;
????????????????es?
= ?e.ErrorMessage();
????????????????ds?
= ?e.Description();
????????????????CString?c1,?c2;
????????????????c1?
= ?es.operator?char * ();
????????????????c2?
= ?ds.operator?char * ();
????????????????info.Format(
" 錯誤:?%s?%s " ,?c1,?c2);
????????????????AfxMessageBox(info);
????????????????conn
-> RollbackTrans();
????????????????return?
FALSE ;
????????????}
????????????conn
-> CommitTrans();
????????}

????????AfterUpdate();

????????return?
TRUE ;
????}

????
int ?GetRecordCount()?{
????????return?m_DataSet.size();
????}

????size_t?AddRow(_DataRow
* ?pnew)?{
????????m_DataSet.push_back(pnew);
????????return?m_DataSet.size();
????}

????void?DelLastRow(size_t?index)?{
????????assert(m_DataSet.size()?
== ?index);

????????
if ?(m_DataSet.size())?{
????????????_DataSet::iterator?it?
= ?m_DataSet.end();
????????????m_DataSet.erase(
-- it);
????????}

????????
????}

????_DataRow
& ?Row( int ?index)?{
????????
int ?i;
????????_DataSet::iterator?it?
= ?m_DataSet.begin();
????????
for ?(i? = ? 0 ;?i? < ?index;?i ++ )?{
????????????it
++ ;
????????}
????????return?
* ( * it);
????}

private :

????void?BeforeUpdate()?{

????}
????void?AfterUpdate()?{
????????
if ?(m_DataSet.size())?{
????????????_DataSet::iterator?it;
????????????it?
= ?m_DataSet.begin();
????????????
while ?(it?! = ?m_DataSet.end())?{
????????????????
if ?(( * it) -> m_en_State? == ?_DataRow::_en_Row_Add?||?( * it) -> m_en_State? == _DataRow::_en_Row_Update)?{
????????????????????(
* it) -> m_en_State? = ?_DataRow::_en_Row_NoChange;
????????????????????it
++ ;
????????????????????continue;
????????????????}
????????????????
// 更新成功,刪除條目
????????????????
if ?(( * it) -> m_en_State? == ?_DataRow::_en_Row_Del)?{
????????????????????_DataSet::iterator?tt;
????????????????????tt?
= ?it ++ ;
????????????????????m_DataSet.erase(tt);
????????????????????continue;
????????????????}
????????????????it
++ ;
????????????}
????????}
????}


????CString?MakeInsertSql(_DataRow
& ?row)?{
????????CString?ret?
= ?CString();
????????
int ?i;
????????ret?
= ? " INSERT?INTO? " ;
????????ret?
+= ?m_TableName.c_str();
????????ret?
+= ? " ?( " ;
????????
for ?(i? = ? 0 ;?i? < ?en_field_count;?i ++ )?{
????????????
if ?(i? > ? 0 )?{
????????????????ret?
+= ? " ,? " ;
????????????}
????????????ret?
+= ?m_FieldNames.at(i).m_FieldName.c_str();
????????}
????????ret?
+= ? " )?values?( " ;
????????ret?
+= ?_data_get < en_field_count > ::MakeFieldValue(row);
????????ret?
+= ? " ); " ;
????????return?ret;
????}
????BOOL?IsKeyField(
int )?{
????????return?
FALSE ;
????}

????CString?MakeUpdateSql(_DataRow
& ?row)?{
????????CString?ret?
= ?CString();
????????ret?
= ? " UPDATE? " ;
????????ret?
+= ?m_TableName.c_str();
????????ret?
+= ? " ?SET? " ;
????????ret?
+= ?_data_get < en_field_count > ::MakeUpdateValue(row,?m_FieldNames,?m_TableName.c_str());
????????ret?
+= ? " ; " ;
????????return?ret;
????}

????CString?MakeDeleteSql(_DataRow
& ?row)?{
????????CString?ret?
= ?CString();
????????ret?
= ? " Delete?*?from? " ;
????????ret?
+= ?m_TableName.c_str();
????????ret?
+= ? " ? " ;
????????ret?
+= ?_data_get < en_field_count > ::MakeDeleteValue(row,?m_FieldNames);
????????ret?
+= ? " ; " ;
????????return?ret;
????}
public :
????void?Test()?{
????????_DataSet::iterator?it?
= ?m_DataSet.begin();
????????MakeInsertSql(
* ( * it));
????????MakeUpdateSql(
* ( * it));
????????MakeDeleteSql(
* ( * it));
????}

private :

????BOOL?GetFieldName(CCom_Recordset
& ?reset)?{
????????
long ?fieldcount;
????????
long ?i;
????????CComFields?lfields;
????????lfields?
= ?reset -> GetFields();
????????fieldcount?
= ?lfields -> GetCount();
????????m_FieldNames.clear();
????????
for (i? = 0 ;?i? < ?fieldcount;?i ++ )?{
????????????CComField?lf;
????????????lf?
= ?lfields -> GetItem(_variant_t(i));
????????????_bstr_t?fname?
= ?lf -> Name;
????????????CString?fieldname??
= ?fname.operator?char * ();
????????????m_FieldNames.push_back(__tagFieldInfo(std::
string (LPCTSTR(fieldname)),?lf -> Type));
????????????TRACE(
" fieldname?%s\n " ,?fieldname);
????????}
????????return?
TRUE ;
????}

????virtual?BOOL?Save()?{
????????BOOL?ret?
= ? FALSE ;
????????
for (_DataSet::iterator?it? = ?m_DataSet.begin();?it?! = ?m_DataSet.end();?it ++ )?{
????????}
????????return?ret;
????}

????BOOL?InsertRow()?{
????????return?
0 ;
????}

????BOOL?DeleteRow()?{
????????return?
0 ;
????}

????BOOL?UpdateRow()?{
????????return?
0 ;
????}

public :
????void?SetTableName(LPCTSTR?table_name)?{
????????m_TableName?
= ?table_name;
????}

????void?SetKey(
int ?index)?{
????????m_FieldNames.at(index).m_KeyField?
= ? TRUE ;
????}

????void?SetKey(
int ?id1,? int ?id2)?{
????????m_FieldNames.at(id1).m_KeyField?
= ? TRUE ;
????????m_FieldNames.at(id2).m_KeyField?
= ? TRUE ;
????}

????void?Clear()?{
????????
if ?(m_DataSet.size())?{
????????????_DataSet::iterator?it;
????????????
for ?(it? = ?m_DataSet.begin();?it?! = ?m_DataSet.end();?it ++ )?{
????????????????delete?
* it;
????????????}
????????????m_DataSet.clear();
????????}
????}

????void?SetModified()?{
????????m_Modified?
= ? true ;
????}

????bool?GetModified()?{
????????return?m_Modified;
????}


public :
????data_op_recordset()?:?m_Modified(
false )?{}
????~data_op_recordset()?{
????????Clear();
????}
protected:
private :
????_DataSet?m_DataSet;
????_FieldNames?m_FieldNames;
????std::
string ?m_TableName;
????bool?m_Modified;
};


例子代碼在:

http://m.shnenglu.com/Files/hdqqq/ado_template.rar

在其中實現(xiàn)了表格數(shù)據(jù)顯示,添加記錄和一個跨表數(shù)據(jù)查詢結(jié)果的顯示.

vc 6 工程, 編譯需要loki 和boost 庫支持.

過完春節(jié)以后,很多事情,一直沒有時間寫東西,這次把基于模板的數(shù)據(jù)操作類寫完,總算是對自己有個交待.


?

posted on 2007-03-26 11:18 hdqqq 閱讀(1415) 評論(3)  編輯 收藏 引用 所屬分類: c/c++

評論

# re: 創(chuàng)建一個基于模板的數(shù)據(jù)庫記錄集操作類(四) 2007-08-02 10:52 bbp
想說的是,你的代碼依賴于Windows平臺,不好!  回復(fù)  更多評論
  

# re: 創(chuàng)建一個基于模板的數(shù)據(jù)庫記錄集操作類(四)[未登錄] 2007-08-28 09:16 hdqqq
跨平臺的數(shù)據(jù)庫操作接口,比較難實現(xiàn).  回復(fù)  更多評論
  

# re: 創(chuàng)建一個基于模板的數(shù)據(jù)庫記錄集操作類(四) 2008-09-22 10:40 Arcol
沒看你文章之前,我也寫了一份類似的用模板封裝數(shù)據(jù)庫ADO接口,同樣是基于TypeList思想實現(xiàn)的,感覺很有意思,親自寫過一次才知道難度有多大

另外,要駁一下上面的回帖,跨平臺的不一定就是好東西,本來封裝的就是ado數(shù)據(jù)庫接口,是否跨平臺已經(jīng)沒多大意義了  回復(fù)  更多評論
  

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美久久久久久| 国产精品免费福利| 亚洲精品老司机| 噜噜噜躁狠狠躁狠狠精品视频 | 亚洲裸体俱乐部裸体舞表演av| 极品尤物av久久免费看| 国产亚洲精品v| 影音欧美亚洲| 亚洲精品在线观看免费| 这里只有精品电影| 欧美一区免费| 免费亚洲视频| 日韩午夜剧场| 久久国产日韩| 欧美精品午夜视频| 国产视频自拍一区| 亚洲精品欧美日韩| 欧美一区综合| 亚洲国产精品传媒在线观看 | 国产一区二区日韩精品欧美精品| 一区二区在线视频观看| 亚洲私人影吧| 国产日韩精品电影| 伊人夜夜躁av伊人久久| 一本色道久久综合亚洲精品不| 午夜精品免费在线| 亚洲国产成人久久综合| 亚洲欧美高清| 欧美精品在线一区| 好吊色欧美一区二区三区四区 | 欧美激情综合在线| 韩国精品久久久999| 亚洲综合国产| 亚洲激情偷拍| 久久漫画官网| 国产一区二区你懂的| 亚洲一二三区在线| 亚洲黄色天堂| 久久久久久久999精品视频| 国产精品久久久一本精品| av成人老司机| 亚洲电影自拍| 久久久久www| 国产精品乱码一区二区三区| 亚洲乱码国产乱码精品精可以看 | 亚洲精品美女久久久久| 久久午夜精品| 欧美在线一二三四区| 夜夜嗨av一区二区三区网站四季av| 午夜免费在线观看精品视频| 欧美日韩午夜精品| 亚洲国产另类 国产精品国产免费| 久久狠狠婷婷| 国产人久久人人人人爽| 亚洲第一色中文字幕| 欧美一区二区三区视频在线| 一本色道久久综合精品竹菊| 久久视频一区| 亚洲激情在线观看视频免费| 在线一区二区三区做爰视频网站| 久久久精品五月天| 欧美高清日韩| 久久精品亚洲国产奇米99| 国产日韩在线亚洲字幕中文| 欧美成人dvd在线视频| 伊人久久婷婷色综合98网| 久久综合久久久久88| 久久久久久久综合| 伊人久久大香线蕉av超碰演员| 久久噜噜亚洲综合| 久久久久久久综合| 亚洲三级色网| 一区二区免费在线观看| 国产精品久久国产三级国电话系列| 亚洲图片在线| 亚洲欧美日韩国产中文| 狠狠色综合网| 亚洲日本va午夜在线影院| 欧美日韩免费| 久久精品国产91精品亚洲| 久久99在线观看| 亚洲区第一页| 亚洲线精品一区二区三区八戒| 国产日本欧美一区二区三区| 久热国产精品视频| 欧美日韩一区二区三区高清| 欧美亚洲视频| 蜜臀av性久久久久蜜臀aⅴ| 一本久久青青| 欧美一区二区三区四区高清 | 亚洲国产三级| 日韩亚洲欧美一区二区三区| 国产精品一香蕉国产线看观看| 久久午夜羞羞影院免费观看| 欧美久久久久久久| 久久久久中文| 欧美日本精品| 久久久久一本一区二区青青蜜月| 欧美 日韩 国产 一区| 午夜日韩福利| 欧美精品91| 久久一区二区三区国产精品| 欧美日韩免费精品| 狼人天天伊人久久| 国产精品一区视频| 免费观看日韩| 亚洲美女视频在线观看| 国产精品久久久久久久久搜平片| 亚洲国产精品久久| 亚洲精品一区二区三区樱花 | 亚洲国产经典视频| 一区二区av| 亚洲高清不卡在线| 亚洲欧美卡通另类91av| 99ri日韩精品视频| 久久久久久久久一区二区| 亚洲图片欧美午夜| 牛夜精品久久久久久久99黑人| 欧美在线观看一区| 久久精品视频在线| 欧美在线首页| 亚洲午夜视频在线| 欧美在线免费| 欧美日韩国产成人| 午夜精品一区二区三区电影天堂| 欧美高清视频一区二区| 一本色道久久综合亚洲精品小说 | 久久一二三四| 亚洲欧美中日韩| 亚洲一区二区三区三| 亚洲美女精品一区| 欧美激情精品久久久久久蜜臀 | 一区二区三区四区在线| 性18欧美另类| 黄色成人在线网址| 美女国产一区| 久久―日本道色综合久久| 亚洲一区在线看| 欧美一区二区日韩| 欧美一区免费| 久久精品亚洲国产奇米99| 亚洲第一福利在线观看| 亚洲国产精品久久91精品| 亚洲激情专区| 国产精品一区二区在线观看不卡| 欧美国产综合视频| 亚洲精品一二区| 亚洲女同性videos| 久热精品在线| 亚洲主播在线观看| 亚洲一区在线视频| 亚洲一二三区在线| 亚洲毛片一区二区| 久久国产一区二区三区| 亚洲精品1234| 欧美一区2区三区4区公司二百| 亚洲网站在线播放| 久久综合网络一区二区| 国产综合色在线| 欧美亚洲在线播放| 欧美激情精品久久久久久久变态| 亚洲免费成人av| 国产精品久久97| 国产毛片精品视频| 久久综合99re88久久爱| 亚洲淫性视频| 欧美剧在线观看| 亚洲一区二区精品在线观看| 久久精品视频网| 亚洲精品国久久99热| 亚洲调教视频在线观看| 国产日本欧美在线观看| 欧美a级片一区| 一本色道久久综合亚洲精品小说| 亚洲欧美三级伦理| 一区二区三区在线免费播放| 欧美男人的天堂| 亚洲电影观看| 国产精品免费区二区三区观看| 久久狠狠婷婷| aa级大片欧美三级| 欧美va天堂在线| 欧美在线看片| 正在播放亚洲| 亚洲激情社区| 黄色亚洲免费| 国产精品久久中文| 欧美理论片在线观看| 久久午夜电影| 欧美一区二区精品久久911| 亚洲毛片av| 免费av成人在线| 久久国产一二区| 午夜精品福利电影| 亚洲精一区二区三区| 国产精品乱人伦一区二区 | 夜夜嗨一区二区| 在线精品福利| 国产在线日韩| 国产亚洲精品久|