0

pathコードは、 がテーブルにあるかどうかを呼び出し元に知らせるだけで済みます。select count(*)以下の c++ コードは、andを使用してこれを正しく実行しますがsqlite3_get_table、短時間だけで終了します。sqlite3_get_table には明らかに問題があります。ここに非常によく似た質問がありますしかし、この答えを私の問題に適用する方法がわかりません。sqlite のドキュメントを読み、sqlite のチュートリアルを読みました。私は sqlite3_exec を使用してデータベースを初期化し、新しいレコードを挿入しましたが、テーブル内で一致するものが見つかった場合または見つからなかった場合に単純に bool を返す堅牢な C++ メソッドを記述する方法がわかりません。私が言ったように、以下のコードは機能しますが、数百回以上の呼び出しでは機能しません。これは、データベースで行う最も単純なことの 1 つではないでしょうか? これを行う方法の簡単な例や、この問題を解決するための最良の方法に関する提案はありますか?

bool ifFound(string path) {
    if (turn_off) return false;
    char **result;
    int nrow = 0; // Number of result rows
    int ncol = 0; // Number of result columns
    char * zErrMsg = 0; // Error message
    if (SQLITE_OK != sqlite3_open(database_path.c_str(), &db)) {
        coutError(__FILE__, __LINE__, "Can't open database " + database_path);
        return false;
    }
    int count = 0;
    string sqlCommand = "select count(*) from my_table where path='"+path+"'";
    if (SQLITE_OK == 
        sqlite3_get_table(db, sqlCommand.c_str(),&result,&nrow,&ncol,&zErrMsg)) 
        count = atoi(result[ncol]);
    else coutError(__FILE__, __LINE__, sqlCommand + " " + zErrMsg);
    sqlite3_free_table(result);
    sqlite3_close(db);
    return (count != 0);
}
4

1 に答える 1

1

sqlite3_get_table を取り除く 1 つの方法は、それを sqlite3_exec、クエリ結果を処理するコールバック、およびカウントを保持するグローバル変数に置き換えることです。

static int pathFound = 0;
int sqlCallback(void *p, int argc, char **argv, char **azColName) {
    pathFound = atoi(argv[0]);
    return 0;
}
void checkPath(string path) {
    string sqlCommand = "select count(*) from m_table where path='"+path+"'";
    rc = sqlite3_exec(db, sqlCommand.c_str(), sqlCallback, 0, &zErrMsg);
    if (SQLITE_OK != rc) {
        coutError(__FILE__, __LINE__, sqlCommand + "\n" + zErrMsg);
        sqlite3_free(zErrMsg);
    }
    sqlite3_close(db);
    if (0 == pathFound) doInsert(path);
}
于 2013-02-12T22:35:07.940 に答える