gpt4 book ai didi

c++ - 完成语句时 SQLite 发生奇怪的崩溃

转载 作者:太空宇宙 更新时间:2023-11-04 14:26:43 27 4
gpt4 key购买 nike

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

我用它反复读取一些数据,第四次,在完成语句时,它崩溃了。我确信我正在将每个准备好的语句与一个定稿匹配。第4次,死机前,数据已经读取成功。

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

我重新运行程序,地址又是一样的。

这是我类(class)的重要部分:

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 行(我舍弃了其余行),并且一个整数作为第一列。在我的机器中,第 3 次迭代崩溃。

注意:从 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 的使用替换为它的内联功能,就不会出现崩溃!!而且地址始终相同。

最佳答案

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

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

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

我已经很多年没有使用指针了,它显示了!感谢您的关注!

编辑:很容易解决。我只是放弃了持有那个引用。调用 SQLiteGateway::select() 的人拥有生成的 SQLiteResultSet 对象并负责删除它。

关于c++ - 完成语句时 SQLite 发生奇怪的崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3342943/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com