根本原因是 std::ifstream 默认按字节流打开,不识别 BOM,将 UTF-8 或 UTF-16 BOM 当普通字符读入导致解析失败;需二进制模式打开、手动检测并跳过 BOM,再转码处理。读取时直接遇到乱码或空内容根本原因是 C++ 标准库的 std::ifstream 默认按字节流打开,不识别 BOM;遇到 UTF-8 BOM(0xEF 0xBB 0xBF)或 UTF-16 BOM(0xFF 0xFE 或 0xFE 0xFF)时,会把它们当普通字符读入,导致后续解析失败------比如 std::getline 读出首行含不可见字节,JSON 解析器直接报 invalid character。实操建议:立即学习"C++免费学习笔记(深入)";用二进制模式打开文件:std::ifstream f("file.txt", std::ios::binary),避免系统层面对换行符或编码的隐式转换手动读前 3 字节判断 UTF-8 BOM,前 2 字节判断 UTF-16 BOM;别依赖 std::codecvt_utf8(已弃用)或平台 API 做自动探测UTF-16 小端和大端 BOM 必须区分:读到 0xFF 0xFE 是 UTF-16LE,0xFE 0xFF 是 UTF-16BE;混用会导致每两个字节被颠倒解析std::wifstream 读 UTF-16 文件却只读出一半字符这是最典型的坑:Windows 下 std::wifstream 默认用本地宽字符编码(通常是 UTF-16LE),但若没显式指定 locale,它不会跳过 BOM,也不会按 UTF-16 单位解析------而是把每个字节当一个 wchar_t 处理,结果一个汉字变成两个乱码宽字符。实操建议:立即学习"C++免费学习笔记(深入)";必须在打开前设置 locale:f.imbue(std::locale(f.getloc(), new std::codecvt_utf16<char16_t std::little_endian>));</char16_t>改用 char16_t 流而非 wchar_t 流,避免 Windows 上 wchar_t 是 16 位但语义模糊的问题更稳妥的做法是:先用 std::ifstream 读原始字节,检测 BOM 后,用 std::mbrtoc16 或 std::from_bytes(C++20)转成 std::u16string跨平台读 UTF-8 文件时 BOM 导致 std::stoi 失败Linux/macOS 下很多工具生成的 UTF-8 文件带 BOM,而 std::stoi、std::stod 等函数遇到开头的 0xEF 0xBB 0xBF 会立即返回 0 并设 failbit,不是抛异常,容易被忽略。 文小言 百度旗下新搜索智能助手,有问题,问小言。
相关推荐
Rick19935 分钟前
联合索引是按顺序排好序的步十人9 分钟前
【Redis】网络高并发模型我是一颗柠檬11 分钟前
【Redis】列表与集合Day4(2026年)AOwhisky13 分钟前
Ceph系列第三期:Ceph 集群核心配置与管理weixin_4684668524 分钟前
机器学习之决策树新手实战指南陈天伟教授25 分钟前
安装 AutoCAD 时,“可选工具“ 的详细说明。zcn12631 分钟前
举一反三思路思考形如(列=参数 or decode函数)Hesionberger32 分钟前
巧用异或找出唯一数字(多解)hef28835 分钟前
Python内置函数从入门到实战:list、open等核心用法全解析七老板的blog37 分钟前
【Agent智能体】 任务规划工作流