在最终确定语句时,SQLite发生了奇怪的崩溃

时间:2010-07-27 11:07:03

标签: c++ sqlite

我正在用SQLite编写一个C ++测试应用程序,为此我做了一个小包装器。

我正在反复阅读一些数据,第四次,在完成声明时,它会崩溃。我确定我将每个准备好的声明与最终结果相匹配。第四次,在崩溃之前,数据已成功读取。

一个奇怪的事情:我现在正在显示返回的语句指针(通过printf(“%p”)),在第一次读取时,它总是0048D228。崩溃的是66676767。

我重新运行程序,地址再次相同。

这是我班级的重要部分:

class SQLiteResultSet
{
private:
    sqlite3_stmt *stmt;
public:
    inline SQLiteResultSet(sqlite3_stmt *pStmt) : stmt(pStmt) {}
    inline bool next() { return sqlite3_step(stmt) == SQLITE_ROW; }
    inline const unsigned char *textColumn(int colNum) { 
        return sqlite3_column_text(stmt, colNum); 
    }
    inline const int intColumn(int colNum) {
        return sqlite3_column_int(stmt, colNum); 
    }
    inline void close() { 
        if(isOpen()) { 
            printf("----closing %p\n",stmt);
            sqlite3_finalize(stmt); 
            printf("----closed\n");
            stmt = NULL; 
        } 
    }
    inline bool isOpen() { return stmt != NULL; }
    inline ~SQLiteResultSet() { close(); }
};

class SQLiteGateway
{
private:
    sqlite3 *db;
    SQLiteResultSet *rs;

    void resetResultSet();

public:
    SQLiteGateway(char *dbName);
    SQLiteResultSet *select(char *query);
    bool tableExists(char *tableName);
    void execute(char *query);
public:
    ~SQLiteGateway(void);
};

SQLiteGateway::SQLiteGateway(char *dbName)
{
    db = NULL;
    rs = NULL;
    int error = sqlite3_open(dbName, &db);
    if (error)
    {
        throw sqlite3_errmsg(db);
    }
}

void SQLiteGateway::resetResultSet()
{
    if(rs != NULL)
    {
        delete rs;
        rs = NULL;
    }
}

SQLiteResultSet *SQLiteGateway::select(char *query)
{
    sqlite3_stmt    *res;
    const char      *tail;

    resetResultSet();

    int error = sqlite3_prepare_v2(db,query,-1,&res,&tail);
    if (error != SQLITE_OK || res == NULL)
    {
        throw sqlite3_errmsg(db);
    }

    rs = new SQLiteResultSet(res);
    return rs;
}

SQLiteGateway::~SQLiteGateway(void)
{
    resetResultSet();

    if(db != NULL)
    {
        sqlite3_close (db); 
    }
}

编辑:测试程序!

查询应返回至少1行(我丢弃其余的行),并返回一个整数作为第一列。在我的机器中,第三次迭代崩溃。

注意:从SQLiteGateway的声明中删除tableExists()和execute(),我没有在这里复制它们的主体,并且它们没有在这个测试中使用。

int main()
{
    SQLiteGateway *sqlg = NULL;
    try
    {
        sqlg = new SQLiteGateway("./apptest.db");

        for(int i=0; i<5; i++)
        {
            SQLiteResultSet *rs = sqlg->select("select x from mytable");
            if(rs->next()) printf("%d\n", rs->intColumn(0));
            delete rs;
        }
    }
    catch(char *str)
    {
        printf("Abnormal termination: %s\n", str);
    }
    if(sqlg != NULL) 
    {
        delete sqlg;
    }
}

这是我通过测试获得的输出(0是读取的值,并且在第3次迭代时不显示 - 关闭/关闭的行在完成之前/之后)。

0
----closing 0048D1E0
----closed
0
----closing 0048D1E0
----closed
----closing 0048DFF8

我仍然无能为力但是我发现,在这个测试中,如果我用内联替换SQLiteGateway的使用,那就没有崩溃!地址总是一样的。

1 个答案:

答案 0 :(得分:0)

我发现了错误:这就是我如何管理SQLiteResultSet对象的生命周期!

SQLiteGateway持有生成的最后一个结果集的引用,如果它不为null,则在下一个查询之前删除它。但是我可能已经将其删除了,这实际上就是我正在做的事情!结果集对象不再存在,但SQLiteGateway对象不知道,它仍然有一个非空指针。

我必须重新设计解决方案。

多年来我一直没有使用指针,它显示了!谢谢你的兴趣!

编辑:轻松解决。我只是放弃了那个参考。调用SQLiteGateway :: select()的人拥有生成的SQLiteResultSet对象,并负责删除它。