SQLiteのC言語APIを使ってprepared statementで、BLOBを使ってみます。
基本的には、以下のサイトに書いてある方法ですが、日本語で書いてみることにします。 http://www.sqlite.org/capi3ref.html#sqlite3_stmt
- sqlite3_prepare_v2()(あるいはそれに類するもの)を使って、sqlite3_stmtのオブジェクトを作る。
- sqlite3_bind_*() インターフェースを使ってパラメータに値をBind。
- sqlite3_step()を使ってSQL実行。(sqlite3_step()は何回でも実行できる)
- sqlite3_reset()を使ってstatementをリセットして、step 2に戻る。これは何度でもできるし、やらなくても良い。
- sqlite3_finalize()を使ってオブジェクトを破棄する。
というわけで、いきなりサンプルコードw 便利だなーと思ったのは、sqlite3_errmsg()関数。dbを引数にすると、
$ gcc -o sqlite_test sqlite_test.c -lsqlite3
$ chmod 000 test.db
と、DBファイルを読み書き出来ないようにして、実行すると、
$ ./sqlite_test
open error:14, unable to open database file
と、何でエラーになったのかが表示されます。
さらに、このDBファイルをsqlite3コマンドで見たときの話は次回のエントリに書きたいと思います。
#include
#include
#include
#include
struct human {
char name[256];
int age;
int height;
int sex;
};
int main(int argc, char *argv[])
{
struct human human[5] = {
{ "Isono Katsuo", 32, 168, 0 },
{ "Isono Wakame", 29, 158, 1 },
{ "Huguta Tarao", 20, 180, 0 },
{ "Huguta Masuo", 52, 178, 0 },
{ "Huguta Sazae", 48, 161, 1 } };
sqlite3 *db;
sqlite3_stmt *dropStmt = NULL;
sqlite3_stmt *createStmt = NULL;
sqlite3_stmt *selectStmt = NULL;
sqlite3_stmt *insertStmt = NULL;
char *drop_tbl_sql = "drop table member";
char *create_tbl_sql = "create table member"
"(id INTEGER PRIMARY KEY,"
"human BLOB NOT NULL )";
char *insert_tbl_sql = "INSERT INTO member (id, human) values (?, ?)";
char *select_tbl_sql = "select human from member where id = ?";
char *pzTail;
int rc = 0;
int exitcode = 0;
rc = sqlite3_open("test.db", &db);
//rc = sqlite3_open(":memory:", &db);
if (rc != SQLITE_OK) {
printf("open error:%d, %s\n", rc, sqlite3_errmsg(db));
exitcode = 1;
goto end;
}
/* cleanup. ignore error. */
sqlite3_prepare_v2(db, drop_tbl_sql, -1, &dropStmt, NULL);
sqlite3_step(dropStmt);
rc = sqlite3_prepare_v2(db, create_tbl_sql, -1, &createStmt, NULL);
if (rc != SQLITE_OK) {
printf("create error:%d, %s\n", rc, sqlite3_errmsg(db));
exitcode = 1;
goto end;
}
rc = sqlite3_step(createStmt);
if (rc != SQLITE_DONE) {
printf("create error:%d, %s\n", rc, sqlite3_errmsg(db));
exitcode = 1;
goto end;
}
rc = sqlite3_prepare_v2(db, insert_tbl_sql, -1, &insertStmt, NULL);
if (rc != SQLITE_OK) {
printf("insert error:%d, %s\n", rc, sqlite3_errmsg(db));
exitcode = 1;
goto end;
}
rc = sqlite3_prepare_v2(db, select_tbl_sql, -1, &selectStmt, NULL);
if (rc != SQLITE_OK) {
printf("select error:%d, %s\n", rc, sqlite3_errmsg(db));
exitcode = 1;
goto end;
}
int i = 0;
for (i = 0; i < 5; i++) {
sqlite3_reset(insertStmt);
sqlite3_bind_int(insertStmt, 1, i);
sqlite3_bind_blob(insertStmt, 2, &human[i], sizeof(struct human), SQLITE_STATIC);
rc = sqlite3_step(insertStmt);
if (rc != SQLITE_DONE) {
printf("insert error:%d, %s\n", rc, sqlite3_errmsg(db));
exitcode = 1;
goto end;
}
}
printf("-------------------------------------\n");
for (i = 0; i < 5; i++) {
sqlite3_reset(selectStmt);
sqlite3_bind_int(selectStmt, 1, i);
rc = sqlite3_step(selectStmt);
//printf("select result:%d\n", rc);
if (rc == SQLITE_ROW) {
struct human *t = (struct human*)sqlite3_column_blob(selectStmt, 0);
printf("ID: %d\n\t name: %s\n\t age: %d\n\t height: %d\n\t sex: %s\n",
i, t->name, t->age, t->height, t->sex == 0 ? "M" : "F");
printf("-------------------------------------\n");
}
}
end:
sqlite3_finalize(selectStmt);
sqlite3_finalize(insertStmt);
sqlite3_finalize(createStmt);
sqlite3_finalize(dropStmt);
return exitcode;
}
Comments
comments powered by Disqus