VC++连接MySQL数据库 常用的方式有三种:ADO、mysql++,mysql API ; 本文只讲述ADO的连接方式。

      为什么要使用连接池? 对于简单的数据库应用,完全可以先创建一个常连接(此连接永远不关闭,直接数进程退出),但是这样做至少会引起两个问题:(1)资源竞争,多个数据库请求操作不能同时进行,后一请求必须要等到前一请求完成后才能进行;(2)多线程情况下容易出现混乱,甚至出现资源异常释放。还有一种方法,就是使用数据库时创建连接,使用完后关闭连接回收资源。这种方式在数据库操作频繁的情况下会出现严重的效率问题。

数据库连接池

     百度百科给出的解释说明如下:

  数据库连接池负责分配、管理和释放数据库连接,它允许应用程序重复使用一个现有的数据库连接,而不是再重新建立一个;释放空闲时间超过最大空闲时间的数据库连接来避免因为没有释放数据库连接而引起的数据库连接遗漏。这项技术能明显提高对数据库操作的性能。

   使用数据库连接池至少带来以下几个好处:

  1、资源复用

    数据库连接得到复用,避免了频繁创建、释放引起的系统性能开销。减少了内存碎片以及数据库线程(甚至是进程)的数量。

 2、提高系统响应速度

     由于数据库连接资源得到复用,这毫无疑问会提高系统的整体响应速度。

3、避免资源泄漏

    所有的连接都集中在连接池中统一管理,这可以避免使用单一连接带来的两个问题。

实现原理

  一个较为完备的数据库连接池应具备以下几个条件:

    (1)实始化时创建一定数据量的连接对象放于连接池中。

    (2)连接池对象要有上限。

    (3)连接使用完毕后要放回连接池而不是直接释放掉。

    (4)长期处于空闲态的连接要释放。

最为完整的实现原理请参考百度百科:数据库连接池。

   下面给出一个简单的ADO数据库连接池实现代码:

   (说明:以下代码没有考虑到上述原理的第(4)点,读者请根据自身需要自行实现之。)

1 //==================头文件 =====================//
  2 //定义数据库连结基本信息结构
  3 typedef struct
  4 {
  5     char    db_ip[20]; //ip地址
  6     uint32  db_port;   //端口
  7     char    db_user[20];//用户
  8     char    db_pwd[32];//密码
  9     char    db_dbname[32];//数据库名
 10 }vos_dbxmlinfo_stru;
 11 
 12 
 13 
 14 class CXCADOPOOL
 15 {    
 16 protected:
 17     CXCADOPOOL();
 18 
 19 public:
 20     virtual ~CXCADOPOOL(void);
 21 
 22     //接口
 23 public:
 24     void    InitConnection(const int iMin, const int iMax);
 25     bool    ExcuteSql(_bstr_t bSql, bool bCheck = true);
 26     bool    GetRecordSet(_bstr_t bSql, _RecordsetPtr& pRecord, long lOption = adCmdText, bool bCheck = true);
 27 
 28 
 29     bool    GetItemValue(_RecordsetPtr pRecord, long nIndex, int& nValue);
 30     bool    GetItemValue(_RecordsetPtr pRecord, long nIndex, UINT64& unValue);
 31     bool    GetItemValue(_RecordsetPtr pRecord, long nIndex, string& strValue);
 32     bool    GetItemValue(_RecordsetPtr pRecord, long nIndex, double& fValue);
 33     bool    GetItemValue(_RecordsetPtr pRecord, long nIndex, float& fValue);
 34     bool    GetItemValue(_RecordsetPtr pRecord, long nIndex, ULONG & nValue);
 35 
 36     bool    GetItemValue(_RecordsetPtr pRecord, long nIndex, short& nValue);
 37     bool    GetItemValue(_RecordsetPtr pRecord, long nIndex, unsigned char& nValue);
 38     bool    GetItemValue(_RecordsetPtr pRecord, string fieldname, string& strValue);
 39 
 40     template<class T>
 41     bool    GetItemValue(_RecordsetPtr pRecord, string fieldname, T& tValue);
 42 
 43     static CXCADOPOOL *Instance();
 44     _ConnectionPtr *GetTransConnection();
 45     void    SendTransCompMsg(_ConnectionPtr *pConptr);
 46     bool    ExecuteTransSql(_ConnectionPtr *pConptr, _bstr_t bSql);
 47 private:
 48     bool    CreateDBConnection(_ConnectionPtr & conptr); //返回一个连接 
 49     void    GetConnectionString(string &strConnect);
 50     _ConnectionPtr * GetConnectionPtr();
 51     void    ReleaseConnectionPtr(_ConnectionPtr &conptr);
 52     void    InitDBConfig();
 53     bool    ExcuteWithoutCheck(_ConnectionPtr &conptr, _bstr_t bSql);
 54     bool    GetRecordSetWithoutCheck(_ConnectionPtr &conptr, _bstr_t bSql, _RecordsetPtr& pRecord, long lOption = adCmdText);
 55     static DWORD WINAPI IdleConnThreadFunc(LPVOID lParam);
 56 private:
 57     
 58     queue<_ConnectionPtr *> m_qConn;
 59     int m_MinConNum;    //最小连接数
 60     int m_MaxConNum;    //最大连接数
 61     int m_CurrentNum;    //当前连接数
 62 
 63     HANDLE          m_Mutex;
 64     HANDLE            m_hEvent;
 65     HANDLE            m_hThread;
 66     DWORD            m_dwThreadId;
 67     HANDLE            m_hThreadEvent;
 68     string                m_strConnect;
 69     static CXCADOPOOL* _instance;
 70 public:
 71     vos_dbxmlinfo_stru    m_stDBInfo;
 72 
 73 };
 74 
 75 template<class T>
 76 bool CXCADOPOOL::GetItemValue(_RecordsetPtr pRecord, string fieldname, T& tValue)
 77 {
 78     try
 79     {
 80         ASSERT_RECORDSET(pRecord);
 81         _variant_t vart = pRecord->GetCollect(_variant_t(fieldname.c_str()));
 82         (tValue = (T)(vart));
 83     }
 84     catch (_com_error &)
 85     {
 86         return false;
 87     }
 88     return true;
 89 }
 90  extern CXCADOPOOL *pAdoPool;
 91 //===================.CPP文件=====================//
 92 
 93 bool CXCADOPOOL::GetItemValue( _RecordsetPtr pRecord, long nIndex, int& nValue )
 94 {
 95     try
 96     {
 97         ASSERT_RECORDSET(pRecord);
 98 
 99         nValue = (int)(pRecord->GetFields()->GetItem(nIndex)->Value);
100     }
101     catch (_com_error &)
102     {
103         return false;
104     }
105     return true;
106 }
107 
108 bool  CXCADOPOOL::GetItemValue(_RecordsetPtr pRecord, long nIndex, UINT64& unValue)
109 {
110     try
111     {
112         ASSERT_RECORDSET(pRecord);
113 
114         unValue = (UINT64)pRecord->GetFields()->GetItem(nIndex)->Value;
115     }
116     catch (_com_error &)
117     {
118         return false;
119     }
120     return true;
121 }
122 
123 
124 bool CXCADOPOOL::GetItemValue(_RecordsetPtr pRecord, long nIndex, ULONG& nValue)
125 {
126 
127     try
128     {
129         ASSERT_RECORDSET(pRecord);
130 
131         nValue = (ULONG)pRecord->GetFields()->GetItem(nIndex)->Value;
132     }
133     catch (_com_error &)
134     {
135         return false;
136     }
137     return true;
138 
139 }
140 bool CXCADOPOOL::GetItemValue(_RecordsetPtr pRecord, long nIndex, string& strValue)
141 {
142     try
143     {
144         ASSERT_RECORDSET(pRecord);
145 
146         _variant_t vart = pRecord->GetFields()->GetItem(nIndex)->Value;
147         if (vart.vt == VT_NULL)
148             return true;
149 
150         strValue = (std::string)(bstr_t)vart;
151     }
152     catch (_com_error &)
153     {
154         return false;
155     }
156 
157     return true;
158 }
159 
160 bool CXCADOPOOL::GetItemValue(_RecordsetPtr pRecord, long nIndex, double& fValue)
161 {
162     try
163     {
164         ASSERT_RECORDSET(pRecord);
165 
166         fValue = (double)pRecord->GetFields()->GetItem(nIndex)->Value;
167     }
168     catch (_com_error &)
169     {
170         return false;
171     }
172 
173     return true;
174 
175 }
176 
177 
178 bool CXCADOPOOL::GetItemValue(_RecordsetPtr pRecord, long nIndex, float& fValue)
179 {
180     try
181     {
182         ASSERT_RECORDSET(pRecord);
183 
184         fValue = (float)pRecord->GetFields()->GetItem(nIndex)->Value;
185     }
186     catch (_com_error &)
187     {
188         return false;
189     }
190     return true;
191 }
192 
193 bool CXCADOPOOL::GetItemValue(_RecordsetPtr pRecord, long nIndex, short &sValue)
194 {
195     try
196     {
197         ASSERT_RECORDSET(pRecord);
198         sValue = (short)pRecord->GetFields()->GetItem(nIndex)->Value;
199     }
200     catch (_com_error &)
201     {
202         return false;
203     }
204     return true;
205 }
206 
207 bool CXCADOPOOL::GetItemValue(_RecordsetPtr pRecord, long nIndex, unsigned char& cValue)
208 {
209     try
210     {
211         ASSERT_RECORDSET(pRecord);
212         cValue = (unsigned char)pRecord->GetFields()->GetItem(nIndex)->Value;
213     }
214     catch (_com_error &)
215     {
216         return false;
217     }
218     return true;
219 }
220 
221 
222 CXCADOPOOL *pAdoPool = NULL;
223 
224 CXCADOPOOL *CXCADOPOOL::_instance = NULL;
225 
226 
227 CXCADOPOOL::CXCADOPOOL()
228 {
229 
230     ::CoInitialize(NULL);
231 
232     InitDBConfig();
233     GetConnectionString(m_strConnect);
234     m_Mutex = ::CreateMutex(NULL, FALSE, NULL);
235     m_hEvent = ::CreateEvent(NULL, TRUE, FALSE, NULL);
236     m_CurrentNum = 0;
237 
238     m_hThreadEvent = ::CreateEvent(NULL, FALSE, FALSE, NULL);
239     m_hThread = ::CreateThread(NULL, 0,(LPTHREAD_START_ROUTINE)IdleConnThreadFunc, this, 0, &m_dwThreadId);
240     WaitForSingleObject(m_hThreadEvent, INFINITE);
241     CloseHandle(m_hThreadEvent);
242    
245 }
246 
247 CXCADOPOOL::~CXCADOPOOL(void)
248 {
249     ::CoUninitialize();
250 }
251 
252 void CXCADOPOOL::InitConnection(const int iMin, const int iMax)
253 {
254     static bool bInitial = true;
255     if (bInitial)
256     {
257         m_MinConNum = iMin;
258         m_MaxConNum = iMax;
259         for (int i = 0; i < iMin; i++)
260         {
261             _ConnectionPtr *conptr = new _ConnectionPtr;
262             if (CreateDBConnection(*conptr))
263             {
264                 WaitForSingleObject(m_Mutex,INFINITE);
265                 m_qConn.push(conptr);
266                 m_CurrentNum++;
267                 ReleaseMutex(m_Mutex);
268             }
269         }
270         bInitial = false;
271     }
272 }
273         
274 bool CXCADOPOOL::CreateDBConnection(_ConnectionPtr & conptr)
275 {
276     try
277     {
278         //conptr.CreateInstance("ADODB.Connection");
279         conptr.CreateInstance(__uuidof(Connection));
280     
281         HRESULT hr = conptr->Open(m_strConnect.c_str(), "", "", adModeUnknown);
282         if (FAILED(hr))
283         {
284                         return false;
285         }        
286     }
287     catch (_com_error &e)
288     {
289                 return false;
290     }
291     return true;
292 }
293 
294 void CXCADOPOOL::GetConnectionString(string &strConnect)
295 {
296     USES_CONVERSION;
297     CString str;
298     str.Format(_T("Driver=MySQL ODBC 5.3 Unicode Driver;SERVER=%s;UID=%s;PWD=%s;DATABASE=%s;PORT=%d"),
299         A2T((char*)m_stDBInfo.db_ip), A2T((char*)m_stDBInfo.db_user), A2T((char*)m_stDBInfo.db_pwd), A2T((char*)m_stDBInfo.db_dbname), m_stDBInfo.db_port);
300     strConnect = T2A(str);
301 
302 }
303 
304 void CXCADOPOOL::InitDBConfig()
305 {
306     GetPrivateProfileStringA("DBInfo", "host", "localhost", m_stDBInfo.db_ip, 20, ".\\DB.ini");
307     m_stDBInfo.db_port = GetPrivateProfileIntA("DBInfo", "port", 3306, ".\\DB.ini");
308     GetPrivateProfileStringA("DBInfo", "dbname", "", m_stDBInfo.db_dbname, 32, ".\\DB.ini");
309     GetPrivateProfileStringA("DBInfo", "user", "", m_stDBInfo.db_user, 20, ".\\DB.ini");
310 
311     char pbuf_text[255] = { 0 };
312     GetPrivateProfileStringA("DBInfo", "password", "", pbuf_text, 255, ".\\DB.ini");  
313 }
314 
315 bool CXCADOPOOL::ExcuteSql(_bstr_t bSql, bool bCheck)
316 {
317  
326     _ConnectionPtr *conptr = GetConnectionPtr();
327     bool bExec = ExcuteWithoutCheck(*conptr, bSql);
330     PostThreadMessage(m_dwThreadId, WM_USER_DB_THREAD_MSG, (WPARAM)conptr,NULL);
331     return bExec;
332 }
333 
334 _ConnectionPtr *  CXCADOPOOL::GetConnectionPtr()
335 {
336     //找出空闲连接
337     while (1)
338     {
339         WaitForSingleObject(m_Mutex, INFINITE);
340         _ConnectionPtr *conptr;
341         if (m_qConn.empty())
342         {
343             if (m_CurrentNum < m_MaxConNum)
344             {
345                 conptr = new _ConnectionPtr;
346                 if (CreateDBConnection(*conptr))
347                 {
348                     m_CurrentNum++;
349                 }
350             }
351             else
352             {
353                 //等待连接释放
354                 ResetEvent(m_hEvent);
355                 ReleaseMutex(m_Mutex);
356                 WaitForSingleObject(m_hEvent, INFINITE);
357                 continue;
358             }
359         }
360         else
361         {
362             conptr = m_qConn.front();
363             m_qConn.pop();
364         }
365 
366         ReleaseMutex(m_Mutex);
367         return conptr;
368     }
369     
370     
371 }
372 
373 DWORD WINAPI CXCADOPOOL::IdleConnThreadFunc(LPVOID lParam)
374 {
375     MSG msg;
376     PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE);
377     CXCADOPOOL *pCXCADOPOOL = static_cast<CXCADOPOOL *>(lParam);
378     SetEvent(pCXCADOPOOL->m_hThreadEvent);
379 
380     while (1)
381     {
382         if (GetMessage(&msg, 0, 0, 0))
383         {
384             switch (msg.message)
385             {
386             case WM_USER_DB_THREAD_MSG:
387             {
388                 _ConnectionPtr *conptr = (_ConnectionPtr *) (msg.wParam);
389             
390                 WaitForSingleObject(pCXCADOPOOL->m_Mutex,INFINITE);
391                 pCXCADOPOOL->m_qConn.push(conptr);
392                 ReleaseMutex(pCXCADOPOOL->m_Mutex);
393                 SetEvent(pCXCADOPOOL->m_hEvent);
394                 
395             }
396             default:
397                 break;
398             }
399         }
400     }
401     return 0;
402 }
403 
404 void CXCADOPOOL::ReleaseConnectionPtr(_ConnectionPtr &conptr)
405 {
406     if (conptr != NULL)
407     {
408         conptr->Close(); //关闭连接
409         conptr.Release(); //释放内存
410         conptr = NULL;
411 
412     }
413 
414 }
415 
416 bool CXCADOPOOL::ExcuteWithoutCheck(_ConnectionPtr &conptr, _bstr_t bSql)
417 {
418     int i = 0;
419     while (i < 3)
420     {
421         try
422         {
423             if (0 != i)
424             {
425                 ReleaseConnectionPtr(conptr);
426                 CreateDBConnection(conptr);
427             }
428             ++i;
429             VARIANT nRecordAffected = { 0 };
430             conptr->Execute(bSql, &nRecordAffected, adCmdText);
431             //ReleaseMutex(m_Mutex);
432 
433             if (nRecordAffected.date < 0)
434             {
435                 return false;
436             }
437             break;
438         }
439         catch (_com_error&e)
440         {
441                     }
442         catch (...)
443         {
444                 
445                  }
446     }
447     if (i == 3)
448     {
449        return false;
450     }
451 
452     return true;
453 }
454 
455 bool CXCADOPOOL::GetRecordSet(_bstr_t bSql, _RecordsetPtr& pRecord, long lOption /*= adCmdText*/, bool bCheck)
456 {     465     _ConnectionPtr *conptr = GetConnectionPtr();
466     bool bExec = GetRecordSetWithoutCheck(*conptr, bSql, pRecord,lOption);
467     PostThreadMessage(m_dwThreadId, WM_USER_DB_THREAD_MSG, (WPARAM)conptr, NULL);
468     return bExec;
469 }
470 
471 bool CXCADOPOOL::GetRecordSetWithoutCheck(_ConnectionPtr &conptr, _bstr_t bSql, _RecordsetPtr& pRecord, long lOption /*= adCmdText*/)
472 {
473     for (int i = 0; i < 3; ++i)
474     {
475         try
476         {
477             if (0 != i)
478             {
479                 ReleaseConnectionPtr(conptr);
480                 CreateDBConnection(conptr);
481             }
482             HRESULT hr = pRecord.CreateInstance(__uuidof(Recordset));
483             if (SUCCEEDED(hr))
484             {
485                 pRecord->CursorLocation = adUseClient;
486                 HRESULT ht = pRecord->Open(bSql, _variant_t((IDispatch *)conptr), adOpenDynamic, adLockOptimistic, lOption);
487                 return SUCCEEDED(ht);
488             }
489             return false;
490         }
491         catch (_com_error&e)
492         {        }
493         catch (...)
494         {
495                 }
496     }
497     return false;
498 }
499 
500 bool CXCADOPOOL::GetItemValue(_RecordsetPtr pRecord, string fieldname, string& strValue)
501 {
502     try
503     {
504         ASSERT_RECORDSET(pRecord);
505         _variant_t vart = pRecord->GetCollect(_variant_t(fieldname.c_str()));
506         strValue = (std::string)(bstr_t)vart;
507     }
508     catch (_com_error &)
509     {
510         return false;
511     }
512     return true;
513 }
514 
515 CXCADOPOOL * CXCADOPOOL::Instance()
516 {
517     if (NULL == _instance)
518     {
519         _instance = new CXCADOPOOL;
520     }
521     return _instance;
522 }
523 
524 _ConnectionPtr * CXCADOPOOL::GetTransConnection()
525 {
526     _ConnectionPtr *pConptr = this->GetConnectionPtr();
527     //执行一个查询语句验证下确保当前连接可用
528     if ((*pConptr)->State != adStateOpen)
529     {
530         ReleaseConnectionPtr(*pConptr);
531         CreateDBConnection(*pConptr);
532     }
533     return pConptr;
534 }
535 
536 void CXCADOPOOL::SendTransCompMsg(_ConnectionPtr *pConptr)
537 {
538     PostThreadMessage(m_dwThreadId, WM_USER_DB_THREAD_MSG, (WPARAM)pConptr, NULL);
539 }
540 
541 bool CXCADOPOOL::ExecuteTransSql(_ConnectionPtr *pConptr, _bstr_t bSql)
542 {
543   return ExcuteWithoutCheck(*pConptr, bSql);
544 }
545 
546 
547 
548 
549