SQLite 中文写入失败问题总结

SQLite 中文写入失败问题总结与解决方案

在 Windows 下使用 C++ 操作 SQLite 数据库时,中文字段经常出现 写入成功但内容显示为 BLOB乱码 的问题。根本原因在于 SQLite 要求字符串以 UTF-8 编码 存储,而默认的 std::string 中文通常是 GB2312/ANSI 编码。

🧩 问题现象

在 C++ (如 Visual Studio) 中使用 std::string 把含有中文的字符串插入到 SQLite 数据库时:

  • 插入后数据表中文本列显示为 BLOB
  • 或者表面显示为 乱码,无法正常显示中文

示例

cpp 复制代码
std::string desc = "搬运动作:从 Port2 取片 → Port3 放片";
m_pDB->executeQuery("INSERT INTO ... VALUES ('" + desc + "');");

结果: description 列在 SQLite 中显示为 BLOB,而不是文本


🔍 根本原因

SQLite 要求所有 TEXT 类型的字符串必须为 UTF-8 编码。

而 VS 编译器中直接写入 std::string 的中文,实际编码是 GB2312 / GBK / ANSI,并非 UTF-8

导致结果:

  • SQLite 无法识别非 UTF-8 内容,转而存储成 BLOB

✅ 解决方案

✅ 方法一:使用 ansiToUtf8() 将 ANSI 转换为 UTF-8

cpp 复制代码
// 本地编码转为 UTF-8
std::string TransferManager::ansiToUtf8(const std::string& ansiStr) {
    int wideLen = MultiByteToWideChar(CP_ACP, 0, ansiStr.c_str(), -1, nullptr, 0);
    std::wstring wideStr(wideLen, 0);
    MultiByteToWideChar(CP_ACP, 0, ansiStr.c_str(), -1, &wideStr[0], wideLen);

    int utf8Len = WideCharToMultiByte(CP_UTF8, 0, wideStr.c_str(), -1, nullptr, 0, nullptr, nullptr);
    std::string utf8Str(utf8Len, 0);
    WideCharToMultiByte(CP_UTF8, 0, wideStr.c_str(), -1, &utf8Str[0], utf8Len, nullptr, nullptr);

    utf8Str.pop_back(); // 去掉最后的 '\0'
    return utf8Str;
}

使用:

cpp 复制代码
std::string raw = "搬运动作:从 Port2 取片 → Port3 放片"; // 系统本地编码
std::string utf8 = TransferManager::ansiToUtf8(raw);
data.strDescription = utf8;

✅ 方法二:如果使用了 Unicode 工程(如 std::wstring / CStringW

可使用:

cpp 复制代码
std::string TransferManager::wstringToUtf8(const std::wstring& wstr);

或者使用 CStringWstd::wstringUTF-8

cpp 复制代码
CStringW wdesc = L"搬运动作:从 Port2 取片 → Port3 放片";
data.strDescription = TransferManager::wstringToUtf8(std::wstring(wdesc));

📦 可选补充:从 SQLite 中读取 UTF-8 → 转为本地编码显示

cpp 复制代码
// UTF-8 转为本地编码
std::string TransferManager::utf8ToAnsi(const std::string& utf8Str) {
    int wideLen = MultiByteToWideChar(CP_UTF8, 0, utf8Str.c_str(), -1, NULL, 0);
    std::wstring wideStr(wideLen, 0);
    MultiByteToWideChar(CP_UTF8, 0, utf8Str.c_str(), -1, &wideStr[0], wideLen);

    int ansiLen = WideCharToMultiByte(CP_ACP, 0, wideStr.c_str(), -1, NULL, 0, NULL, NULL);
    std::string ansiStr(ansiLen, 0);
    WideCharToMultiByte(CP_ACP, 0, wideStr.c_str(), -1, &ansiStr[0], ansiLen, NULL, NULL);

    ansiStr.pop_back(); // 去掉最后的 '\0'
    return ansiStr;
}

可用于日志输出或 UI 控件显示。


🧪 实际检测建议

插入前输出:

cpp 复制代码
printf("准备插入 UTF-8 字符串:%s\n", data.strDescription.c_str());

验证方式:

  • SQLite 中字段类型应为 TEXT,不是 BLOB
  • DB Browser 应正常显示中文,不是问号或乱码

🧠 最佳实践

场景 建议方式
插入中文到 SQLite ansiToUtf8
从 SQLite 读取后展示 utf8ToAnsi
项目编码设置 设置为 UTF-8 无 BOM
VS 源文件保存编码 另存为 UTF-8 编码

✅ 总结

只要 std::string 中的中文不是 UTF-8 编码,SQLite 插入后就会出现异常(显示为 BLOB 或乱码)。

相关推荐
程序喵大人14 小时前
SQLITE问题整理
开发语言·数据库·c++·sqlite
计算机程序设计小李同学1 天前
基于Python的在线零食购物商城系统的设计与实现
数据库·sqlite
openinstall全渠道统计2 天前
开发者指南:广告投放系统搭建与前后端数据打通全流程
windows·git·oracle·eclipse·sqlite·github
Sammyyyyy3 天前
Django 6.0 发布,新增原生任务队列与 CSP 支持
数据库·后端·python·django·sqlite·servbay
赤龙绕月4 天前
SQLite NET
数据库·sqlite
透明的玻璃杯5 天前
sqlite数据库链接池二
数据库·oracle·sqlite
透明的玻璃杯5 天前
sqlite数据库连接池
jvm·数据库·sqlite
越甲八千6 天前
ASGI和AWSIG区别
数据库·python·sqlite
码农12138号6 天前
Bugku 2023 HackINI Virtual Shop 和 2023 HackINI Virtual Shop 2
web安全·sqlite·sql注入
飞Link6 天前
【Django】Django 调用外部 Python 程序的完整指南
后端·python·django·sqlite