亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 數據庫 > SQLite > 正文

用sqlite執行標準SQL語法

2024-09-07 00:10:05
字體:
來源:轉載
供稿:網友

1. 執行sql語句

int sqlite3_exec(sqlite3*, const char *sql, sqlite3_callback, void *, char **errmsg );
這就是執行一條 sql 語句的函數。
第1個參數不再說了,是前面open函數得到的指針。說了是關鍵數據結構。
第2個參數const char *sql 是一條 sql 語句,以/0結尾。
第3個參數sqlite3_callback 是回調,當這條語句執行之后,sqlite3會去調用你提供的這個函數。
第4個參數void * 是你所提供的指針,你可以傳遞任何一個指針參數到這里,這個參數最終會傳到回調函數里面,這個指針比較重要,可以用來作參數的傳遞。如果不需要傳遞指針給回調函數,可以填NULL。等下我們再看回調函數的寫法,以及這個參數的使用。
第5 個參數char ** errmsg 是錯誤信息。注意是指針的指針。sqlite3里面有很多固定的錯誤信息。執行 sqlite3_exec 之后,執行失敗時可以查閱這個指針(直接 printf(“%s/n”,errmsg))得到一串字符串信息,這串信息告訴你錯在什么地方。sqlite3_exec函數通過修改你傳入的指針的指 針,把你提供的指針指向錯誤提示信息,這樣sqlite3_exec函數外面就可以通過這個 char*得到具體錯誤提示。
說明:通常, sqlite3_callback 和它后面的 void * 這兩個位置都可以填 NULL。填NULL表示你不需要回調。比如你做 insert 操作,做 delete 操作,就沒有必要使用回調。而當你做 select 時,就要使用回調,因為 sqlite3 把數據查出來,得通過回調告訴你查出了什么數據。

2.exec 的回調

typedef int (*sqlite3_callback)(void*,int,char**, char**);
你的回調函數必須定義成上面這個函數的類型。下面給個簡單的例子:
// sqlite 每查到一條記錄,就調用一次這個回調
int LoadMyInfo( void * para, int n_column, char ** column_value, char ** column_name )
{
//para是你在 sqlite3_exec 里傳入的 void * 參數
//通過para參數,你可以傳入一些特殊的指針(比如類指針、結構指針),然后在這里面強制轉換成對應的類型(這里面是void*類型,必須強制轉換成你的類型才可用)。然后操作這些數據
//n_column是這一條記錄有多少個字段 (即這條記錄有多少列)
// char ** column_value 是個關鍵值,查出來的數據都保存在這里,它實際上是個1維數組(不要以為是2維數組),每一個元素都是一個 char * 值,是一個字段內容(用字符串來表示,以/0結尾)
//char ** column_name 跟 column_value是對應的,表示這個字段的字段名稱
//這里,我不使用 para 參數。忽略它的存在.
int i;
printf( “記錄包含 %d 個字段/n”, n_column );
for( i = 0 ; i < n_column; i )
{
printf( “字段名:%s ß> 字段值:%s/n”, column_name[i], column_value[i] );
}
printf( “------------------/n“ );
return 0;
}

int main( int , char ** )
{
sqlite3 * db;
int result;
char * errmsg = NULL;
result = sqlite3_open( “c://Dcg_database.db”, &db );
if( result != SQLITE_OK )
{
//數據庫打開失敗
return -1;
}
//數據庫操作代碼
//創建一個測試表,表名叫 MyTable_1,有2個字段: ID 和 name。其中ID是一個自動增加的類型,以后insert時可以不去指定這個字段,它會自己從0開始增加
result = sqlite3_exec( db, “create table MyTable_1( ID integer primary key autoincrement, name nvarchar(32) )”, NULL, NULL, &errmsg );
if(result != SQLITE_OK )
{
printf( “創建表失敗,錯誤碼:%d,錯誤原因:%s/n”, result, errmsg );
}
//插入一些記錄
result = sqlite3_exec( db, “insert into MyTable_1( name ) values ( ‘走路’ )”, 0, 0, &errmsg );

if(result != SQLITE_OK )
{
printf( “插入記錄失敗,錯誤碼:%d,錯誤原因:%s/n”, result, errmsg );
}

result = sqlite3_exec( db, “insert into MyTable_1( name ) values ( ‘騎單車’ )”, 0, 0, &errmsg );
if(result != SQLITE_OK )
{
printf( “插入記錄失敗,錯誤碼:%d,錯誤原因:%s/n”, result, errmsg );
}

result = sqlite3_exec( db, “insert into MyTable_1( name ) values ( ‘坐汽車’ )”, 0, 0, &errmsg );
if(result != SQLITE_OK )
{
printf( “插入記錄失敗,錯誤碼:%d,錯誤原因:%s/n”, result, errmsg );
}

//開始查詢數據庫
result = sqlite3_exec( db, “select * from MyTable_1”, LoadMyInfo, NULL, &errmsg );

//關閉數據庫
sqlite3_close( db );
return 0;
}

通過上面的例子,應該可以知道如何打開一個數據庫,如何做數據庫基本操作。
有這些知識,基本上可以應付很多數據庫操作了。

3. 不使用回調查詢數據庫

上面介紹的 sqlite3_exec 是使用回調來執行 select 操作。還有一個方法可以直接查詢而不需要回調。但是,我個人感覺還是回調好,因為代碼可以更加整齊,只不過用回調很麻煩,你得聲明一個函數,如果這個函數 是類成員函數,你還不得不把它聲明成 static 的(要問為什么?這又是C 基礎了。C 成員函數實際上隱藏了一個參數:this,C 調用類的成員函數的時候,隱含把類指針當成函數的第一個參數 傳遞進去。結果,這造成跟前面說的 sqlite 回調函數的參數不相符。只有當把成員函數聲明成 static 時,它才沒有多余的隱含的this參數)。
雖然回調顯得代碼整齊,但有時候你還是想要非回調的 select 查詢。這可以通過 sqlite3_get_table 函數做到。
int sqlite3_get_table(sqlite3*, const char *sql, char ***resultp, int *nrow, int *ncolumn, char **errmsg );
第1個參數不再多說,看前面的例子。
第2個參數是 sql 語句,跟 sqlite3_exec 里的 sql 是一樣的。是一個很普通的以/0結尾的char *字符串。
第3個參數是查詢結果,它依然一維數組(不要以為是二維數組,更不要以為是三維數組)。它內存布局是:第一行是字段名稱,后面是緊接著是每個字段的值。下面用例子來說事。
第4個參數是查詢出多少條記錄(即查出多少行)。
第5個參數是多少個字段(多少列)。
第6個參數是錯誤信息,跟前面一樣,這里不多說了。
下面給個簡單例子:

int main( int , char ** )
{
sqlite3 * db;
int result;
char * errmsg = NULL;
char **dbResult; //是 char ** 類型,兩個*號
int nRow, nColumn;
int i , j;
int index;

result = sqlite3_open( “c://Dcg_database.db”, &db );

if( result != SQLITE_OK )
{
//數據庫打開失敗
return -1;
}

//數據庫操作代碼
//假設前面已經創建了 MyTable_1 表
//開始查詢,傳入的 dbResult 已經是 char **,這里又加了一個 & 取地址符,傳遞進去的就成了 char ***
result = sqlite3_get_table( db, “select * from MyTable_1”, &dbResult, &nRow, &nColumn, &errmsg );
if( SQLITE_OK == result )
{
//查詢成功
index = nColumn; //前面說過 dbResult 前面第一行數據是字段名稱,從 nColumn 索引開始才是真正的數據
printf( “查到%d條記錄/n”, nRow );

for( i = 0; i < nRow ; i )
{
printf( “第 %d 條記錄/n”, i 1 );
for( j = 0 ; j < nColumn; j )
{
printf( “字段名:%s ß> 字段值:%s/n”, dbResult[j], dbResult [index] );
index; // dbResult 的字段值是連續的,從第0索引到第 nColumn - 1索引都是字段名稱,從第 nColumn 索引開始,后面都是字段值,它把一個二維的表(傳統的行列表示法)用一個扁平的形式來表示
}
printf( “-------/n” );
}
}

//到這里,不論數據庫查詢是否成功,都釋放 char** 查詢結果,使用 sqlite 提供的功能來釋放
sqlite3_free_table( dbResult );

//關閉數據庫
sqlite3_close( db );
return 0;
}

到這個例子為止,sqlite3 的常用用法都介紹完了。
用以上的方法,再配上 sql 語句,完全可以應付絕大多數數據庫需求。
但有一種情況,用上面方法是無法實現的:需要insert、select 二進制。當需要處理二進制數據時,上面的方法就沒辦法做到。下面這一節說明如何插入二進制數據
(2)操作二進制

sqlite 操作二進制數據需要用一個輔助的數據類型:sqlite3_stmt * 。
這 個數據類型記錄了一個“sql語句”。為什么我把 “sql語句” 用雙引號引起來?因為你可以把 sqlite3_stmt * 所表示的內容看成是 sql語句,但是實際上它不是我們所熟知的sql語句。它是一個已經把sql語句解析了的、用sqlite自己標記記錄的內部數據結構。
正因為這個結構已經被解析了,所以你可以往這個語句里插入二進制數據。當然,把二進制數據插到 sqlite3_stmt 結構里可不能直接 memcpy ,也不能像 std::string 那樣用 號。必須用 sqlite 提供的函數來插入。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品美女久久| 91久久在线观看| 亚洲欧美在线一区二区| 久久免费精品视频| 国产一区二区av| 久久久成人的性感天堂| 欧美激情精品久久久久| 俺去亚洲欧洲欧美日韩| 亚洲天堂一区二区三区| 日本在线精品视频| www.亚洲一区| 日韩中文字幕国产精品| 亚洲欧美国产日韩中文字幕| 欧美大片大片在线播放| 日韩高清中文字幕| 欧美国产日韩免费| 国产91精品久久久| 精品自拍视频在线观看| 国产精品尤物福利片在线观看| 欧美日韩在线视频一区| 色香阁99久久精品久久久| 久久影院模特热| 国产精品自产拍在线观看中文| 久久人91精品久久久久久不卡| 少妇av一区二区三区| 国产999精品久久久| 国产91av在线| 欧美性生交xxxxxdddd| 日韩精品视频免费专区在线播放| 国产精品久久久久aaaa九色| 国产97免费视| 久热精品视频在线| 欧美日韩免费看| 亚洲一区制服诱惑| 欧美色视频日本版| 中文字幕综合在线| 深夜福利国产精品| 久久天堂av综合合色| 久久伊人色综合| 69av视频在线播放| 欧美日韩性视频| 日本欧美爱爱爱| 国产视频丨精品|在线观看| 91精品久久久久久久久久久| 日本精品久久中文字幕佐佐木| 日韩成人在线观看| 日本最新高清不卡中文字幕| 国产精品大陆在线观看| 日韩免费在线电影| 大胆人体色综合| 久久人人爽人人爽爽久久| 性欧美xxxx视频在线观看| 亚洲第一精品久久忘忧草社区| 欧美与欧洲交xxxx免费观看| 91精品久久久久久久久不口人| 欧美性猛交xxxx免费看| 亚洲最新av在线网站| 亚洲第一天堂av| 成人啪啪免费看| 亚洲国产精品成人va在线观看| 欧美最猛性xxxx| 亚洲成人在线网| 91精品免费看| 久久香蕉精品香蕉| 亚洲美女喷白浆| 国产欧美精品一区二区三区介绍| 国产精品亚洲视频在线观看| 午夜精品久久久久久99热软件| 中文字幕一精品亚洲无线一区| 久久久亚洲国产| 91人人爽人人爽人人精88v| 91日本视频在线| 欧美一级bbbbb性bbbb喷潮片| 在线观看国产精品日韩av| 成人在线视频网| 欧美亚洲国产成人精品| 国产xxx69麻豆国语对白| 国产日韩av高清| 羞羞色国产精品| 国产一区二区日韩| 97香蕉久久超级碰碰高清版| 国产精品极品美女在线观看免费| 日韩美女在线播放| 夜夜嗨av一区二区三区免费区| 国产精品福利在线| 欧美裸体xxxxx| 欧美日韩高清在线观看| 5252色成人免费视频| 欧美日韩999| 国产一区二区黄| 在线成人激情视频| 在线成人一区二区| 2019亚洲日韩新视频| 国产精品你懂得| 久精品免费视频| 色偷偷噜噜噜亚洲男人的天堂| 国产一区二中文字幕在线看| 亚洲女人天堂成人av在线| 国产精品精品视频一区二区三区| 国产精品免费视频久久久| 日本在线观看天堂男亚洲| 国产成人在线亚洲欧美| 亚洲xxx自由成熟| 亚洲精品自拍偷拍| 亚洲精品98久久久久久中文字幕| 国产欧美最新羞羞视频在线观看| 97精品在线视频| 国产精品大陆在线观看| 国产69精品久久久久9| 久久av在线看| 亚洲欧美日韩图片| 亚洲激情在线视频| 久久精品中文字幕| 亚洲精品成人久久| 欧美日韩视频免费播放| 91九色国产视频| 亚洲成人激情视频| 97人人模人人爽人人喊中文字| 亚洲综合中文字幕在线观看| 久久精品国产亚洲| 国产一区二区香蕉| 国产精品白嫩美女在线观看| 日韩av日韩在线观看| 在线视频欧美日韩精品| 久久久精品国产| 亚洲视屏在线播放| 亚洲国产精品久久精品怡红院| 大桥未久av一区二区三区| 日韩视频在线免费观看| 国产美女久久久| 亚洲精品在线看| 亚洲www永久成人夜色| 波霸ol色综合久久| 日韩国产欧美精品一区二区三区| 欧美精品一区二区三区国产精品| 日韩在线视频网| 26uuu另类亚洲欧美日本一| 日韩av免费看| 日韩在线播放视频| 国产日韩精品在线| 亚洲自拍在线观看| 91chinesevideo永久地址| 亚洲国产高清福利视频| 97在线观看免费| 国产丝袜一区二区三区免费视频| 欧美日韩国产成人| 亚洲免费av片| 欧美在线免费视频| 久久男人资源视频| 成人免费xxxxx在线观看| 日韩精品极品在线观看播放免费视频| 欧美性生交xxxxx久久久| 亚洲精品在线看| 91在线精品播放| 亚洲综合日韩在线| 精品成人av一区| 欧美日韩高清在线观看| 国产精品久久久av久久久| 国产日韩欧美在线视频观看| 在线亚洲国产精品网| 国产精品狼人色视频一区| 久久伊人免费视频| 久久精品免费播放| 欧美丰满少妇xxxx|