sqlite 使用: 03-问题记录:在使用 sqlite3_bind_text 中设置 SQLITE_STATIC 参数时,处理不当造成的字符乱码

用通义千问生成了一个测试代码,发现存入的文本为乱码,记录下问题与解决方式。

1.问题背景:

以下为问题代码片段,主要看 sqlite3_bind_text 调用文本传参的这一句即可:

cpp 复制代码
//saveTimer 的实现
bool DatabaseManager::saveTimer(const Timer& timer) {
    const char* insertSQL = R"(
        INSERT INTO timers (name, description, expire_time, status)
        VALUES (?, ?, ?, ?);
    )";
    
    sqlite3_stmt* stmt;
    int rc = sqlite3_prepare_v2(db_, insertSQL, -1, &stmt, NULL);
    if (rc != SQLITE_OK) {
        std::cerr << "Failed to prepare statement: " << sqlite3_errmsg(db_) << std::endl;
        return false;
    }
    /* 问题代码部分,SQLITE_STATIC ,timer.getName() 返回的是一个临时变量,语句执行完成就销毁了,就会出问题 */
    sqlite3_bind_text(stmt, 1, timer.getName().c_str(), -1, SQLITE_STATIC);
    sqlite3_bind_text(stmt, 2, timer.getDescription().c_str(), -1, SQLITE_STATIC);
    std::string timeStr = timePointToString(timer.getExpireTime());
    sqlite3_bind_text(stmt, 3, timeStr.c_str(), -1, SQLITE_STATIC);
    sqlite3_bind_int(stmt, 4, static_cast<int>(timer.getStatus()));
    
    rc = sqlite3_step(stmt);
    sqlite3_finalize(stmt);
    
    return rc == SQLITE_DONE;
}

2.问题原因与解决

timer.getName()等方法返回的都是临时变量,语句执行完成就销毁了。sqlite3_bind_text传入的指针

变成了野指针,这会导致数据库文件中的名称和描述为以乱码的形式存在。

需要修改将 SQLITE_STATIC 改为 SQLITE_TRANSIENT。

这两个参数的区别是:

  • SQLITE_STATIC:告诉 SQLite 字符串指针在 SQLite 使用期间不会改变,SQLite 不会复制数据
  • SQLITE_TRANSIENT:告诉 SQLite 需要复制数据,因为原始数据可能在 SQLite 使用完之前被释放或修改。
cpp 复制代码
bool DatabaseManager::saveTimer(const Timer& timer) {
    const char* insertSQL = R"(
        INSERT INTO timers (name, description, expire_time, status)
        VALUES (?, ?, ?, ?);
    )";
    
    sqlite3_stmt* stmt;
    int rc = sqlite3_prepare_v2(db_, insertSQL, -1, &stmt, NULL);
    if (rc != SQLITE_OK) {
        std::cerr << "Failed to prepare statement: " << sqlite3_errmsg(db_) << std::endl;
        return false;
    }
    /*
	修改传入的参数为 SQLITE_TRANSIENT
    */
    sqlite3_bind_text(stmt, 1, timer.getName().c_str(), -1, SQLITE_TRANSIENT);
    sqlite3_bind_text(stmt, 2, timer.getDescription().c_str(), -1, SQLITE_TRANSIENT);
    std::string timeStr = timePointToString(timer.getExpireTime());
    sqlite3_bind_text(stmt, 3, timeStr.c_str(), -1, SQLITE_TRANSIENT);
    sqlite3_bind_int(stmt, 4, static_cast<int>(timer.getStatus()));
    
    rc = sqlite3_step(stmt);
    sqlite3_finalize(stmt);
    
    return rc == SQLITE_DONE;
}
相关推荐
whitelbwwww几秒前
C++基础--类型、函数、作用域、指针、引用、文件
开发语言·c++
leaves falling10 分钟前
C/C++ const:修饰变量和指针的区别(和引用底层关系)
c语言·开发语言·c++
tod11316 分钟前
深入解析ext2文件系统架构
linux·服务器·c++·文件系统·ext
不想写代码的星星19 分钟前
C++ 类型萃取:重生之我在幼儿园修炼类型学
c++
比昨天多敲两行20 分钟前
C++11新特性
开发语言·c++
萧行之29 分钟前
FRP 0.62.0 + Mac Ollama 公网穿透部署+排障实录(标准 TOML 格式)
linux·服务器
齐潇宇31 分钟前
文件共享服务器
linux·运维·网络·文件共享
xiaoye-duck34 分钟前
【C++:C++11】核心特性实战:详解C++11列表初始化、右值引用与移动语义
开发语言·c++·c++11
睡一觉就好了。41 分钟前
二叉搜索树
c++
whitelbwwww1 小时前
C++进阶--类和模板
c++