1. 程式人生 > >初識Mysql 連接器的收獲(包含JDBC API最新文檔)以及一些c++的有用技巧

初識Mysql 連接器的收獲(包含JDBC API最新文檔)以及一些c++的有用技巧

spl 調試 api .com c api logs 錯誤 execute sele

為了寫一個看的過去的數據庫大作業,花了我一早上的時間來研究MySql的c++連接器,大概弄懂了一些基礎的東西,寫出來分享一下。下面是我入門的第一段代碼:(讀這篇文章前,如果你還沒有安裝較新的c++/connect,請你閱讀我的隨筆<vs2015下配置MySQL,使之能使用c++連接完美運行 >)。對了,寫這篇文章時,我的連接器版本是:mysql-connector-c++-8.0.11-windows-x86-64bit。

#include <stdlib.h>
#include <iostream>
#include "mysql_connection.h"
#include 
<cppconn/driver.h> #include <cppconn/exception.h> #include <cppconn/resultset.h> #include <cppconn/statement.h> using namespace std; /* c++連接Mysql步驟: ①聲明四個基礎變量: sql::Driver* driver;//驅動器 sql::Connection* con;//連接器 sql::Statement* stmt;//語句對象 sql::ResultSet* ret;//結果對象 ②創建連接:獲取驅動實例,使用驅動的連接方法連接主機上的數據庫,該方法返回一個連接器指針。 ③創建語句對象:使用連接器創建一個語句 ④查詢結果:使用語句對象的查詢(executeQuery)進行查詢,返回結果對象指針
*/ int main(int argc, char* argv[]) { cout << endl; cout << "正在執行 ‘select ‘hello world‘ as _message‘..." << endl; try { sql::Driver* driver; sql::Connection* con; sql::Statement* stmt; sql::ResultSet* ret; //創建連接 driver = get_driver_instance(); con
= driver->connect("tcp://127.0.0.1:3306", "root", "Chen@970710"); //連接mysql數據庫 test /* 兩種方式: ①con = driver->connect("tcp://127.0.0.1:3306/test", "root", "Chen@970710"); ②con = driver->connect("tcp://127.0.0.1:3306", "root", "Chen@970710"); con->setSchema("test"); */ con->setSchema("test"); stmt = con->createStatement(); //插入一條數據 string mystr = "insert into testuser values(26,\"Li\")"; int iQuerycount = stmt->executeUpdate(mystr); cout << "the Update count is:" << iQuerycount << endl; //獲取查詢結果 ret = stmt->executeQuery("select * from testuser"); //ret = stmt->executeQuery("select ‘hello world‘ as _message"); /* resultset.h中的next()函數,可以每次獲取結果集中的一個元組(一行)(如果存在的話),假若想要提取該行某一列的信息,則可用以get為前綴的 方法來獲取。 */ //逐條遍歷結果元組 while(ret->next()) { cout << "the ID is:" << ret->getInt("id") << " and the name is:" << ret->getString("name") << endl; //cout << "\t...mysql replies:" << endl; ////通過列名獲取某列屬性值 //cout << ret->getString("_message") << endl; //cout << "\t...mysql says it again: "; ////通過數字偏移量,1代表第一列 //cout << ret->getString(1) << endl; } //最後,在查詢結束時,總要記得將按此順序釋放內存:結果->語句->連接器 delete ret; delete stmt; delete con; } catch (sql::SQLException& e) { cout << "# err: SQLException in " << __FILE__; cout << "(" << __FUNCTION__ << ")" << __FILE__ << endl; cout << "# err: " << e.what(); cout << "(MySQL error code: " << e.getErrorCode(); cout << ",SQLState: " << e.getSQLState() << ")" << endl; } cout << endl; system("pause"); return EXIT_SUCCESS; } /* 編譯器內置的宏擴展: ANSI C標準中幾個標準預定義宏: __LINE__:在源代碼中插入當前源代碼行號; __FILE__:在源文件中插入當前源文件名; __DATE__:在源文件中插入當前的編譯日期 __TIME__:在源文件中插入當前編譯時間; __STDC__:當要求程序嚴格遵循ANSI C標準時該標識被賦值為1; __cplusplus:當編寫C++程序時該標識符被定義。 */

上邊的註釋很清楚了,但是更有用的東西是這:https://docs.oracle.com/javase/tutorial/jdbc/basics/processingsqlstatements.html 它是官方的JDBC API文檔,很全很棒。

對於上邊那些宏,作為一個c++程序員,那是相當的有用啊,我在程序註釋裏寫了經常用到的一些宏及其含義,只是有一個我需要特別說明一下,那就是__FUNCTION__,它用來獲取當前被調用函數的函數名,怎麽樣,很強大吧,下面是詳細內容:

僅僅為了獲取函數名,就在函數體中嵌入硬編碼的字符串,這種方法單調乏味還易導致錯誤,不如看一下怎樣使用新的C99特性,在程序運行時獲取函數名吧。  

對象反射庫、調試工具及代碼分析器,經常會需要在運行時訪問函數的名稱,直到不久前,唯一能完成此項任務並且可移植的方法,是手工在函數體內嵌入一個帶有該函數名的硬編碼字符串,不必說,這種方法非常單調無奇,並且輕易導致錯誤。本文將要演示怎樣使用新的C99特性,在運行時獲取函數名。  

那麽怎樣以編程的方式從當前運行的函數中得到函數名呢?  

答案是:使用__FUNCTION__ 及相關宏。 引出問題  

通常,在調試中最讓人心煩的階段,是不斷地檢查是否已調用了特定的函數。對此問題的解決方法,一般是添加一個cout或printf()——假如你使用C語言,如下所示:void myfunc(){cout<<"myfunc()"<<endl;//其他代碼}   

通常在一個典型的工程中,會包含有數千個函數,要在每個函數中都加入一條這樣的輸出語句,無疑難過上“蜀山”啊,因此,需要有一種機制,可以自動地完成這項操作。  

獲取函數名  

  作為一個C++程序員,可能經常碰到 __TIME__、__FILE__、__DATE__ 這樣的宏,它們會在編譯時,分別轉換為包含編譯時間、處理的轉換單元名稱及當前時間的字符串。  

在最新的ISO C標準中,如大家所知的C99,加入了另一個有用的、類似宏的表達式__func__,其會報告未修飾過的(也就是未裁剪過的)、正在被訪問的函數名。請註重,__func__不是一個宏,因為預處理器對此函數一無所知;相反,它是作為一個隱式聲明的常量字符數組實現的: 

static const char __func__[] = "function-name";

在function-name處,為實際的函數名。為激活此特性,某些編譯器需要使用特定的編譯標誌,請查看相應的編譯器文檔,以獲取具體的資料。  

有了它,我們可免去大多數通過手工修改,來顯示函數名的苦差事,以上的例子可如下所示進行重寫: 

void myfunc(){cout<<"__FUNCTION__"<<endl;}

 

官方C99標準為此目的定義的 __func__標識符,確實值得大家關註,然而,ISO C++卻不完全支持所有的C99擴展,因此,大多數的編譯器提供商都使用 __FUNCTION__ 取而代之,而 __FUNCTION__ 通常是一個定義為 __func__ 的宏,之所以使用這個名字,是因為它已受到了大多數的廣泛支持。  

在Visual Studio 2005中,默認情況下,此特性是激活的,但不能與/EP和/P編譯選項同時使用。請註重在IDE環境中,不能識別__func__ ,而要用__FUNCTION__ 代替。  

Comeau的用戶也應使用 __FUNCTION__ ,而不是 __func__ 。  C++ BuilderX的用戶則應使用稍稍不同的名字:__FUNC__ 。  GCC 3.0及更高的版本同時支持 __func__ 和__FUNCTION__ 。  

一旦可自動獲取當前函數名,你可以定義一個如下所示顯示任何函數名的函數:

void show_name(const char * name){cout<<name<<endl;}
void myfunc(){show_name(__FUNCTION__);}//輸出:myfunc
void foo(){show_name(__FUNCTION__);}//輸出:foo

因為 __FUNCTION__ 會在函數大括號開始之後就立即初始化,所以,foo()及myfunc()函數可在參數列表中安全地使用它,而不用擔心重載。

簽名與修飾名   

__FUNCTION__ 特性最初是為C語言設計的,然而,C++程序員也會經常需要有關他們函數的額外信息,在Visual Studio 2005中,還支持另外兩種非標準的擴展特性:__FUNCDNAME__ 與 __FUNCSIG__ ,其分別轉譯為一個函數的修飾名與簽名。函數的修飾名非常有用,例如,在你想要檢查兩個編譯器是否共享同樣的ABI時,就可派得上用場,另外,它還能幫助你破解那些含義模糊的鏈接錯誤,甚至還可用它從一個DLL中調用另一個用C++鏈接的函數。在下例中,show_name()報告了函數的修飾名:void myfunc(){show_name( __FUNCDNAME__); //輸出:?myfunc@@YAXXZ}   

一個函數的簽名由函數名、參數列表、返回類型、內含的命名空間組成。假如它是一個成員函數,它的類名和const/volatile限定符也將是簽名的一部分。以下的代碼演示了一個獨立的函數與一個const成員函數簽名間的不同之處,兩個函數的名稱、返回類型、參數完全相同:

void myfunc(){show_name(__FUNCSIG__);} // void __cdecl myfunc(void)
strUCt S{ void myfunc() const {show_name(__FUNCSIG__);} //void __thiscall S::myfunc(void) const ;

__FUNCTION__的原文請看這裏:http://blog.sina.com.cn/s/blog_65d6476a0101aa4t.html

初識Mysql 連接器的收獲(包含JDBC API最新文檔)以及一些c++的有用技巧