- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在从一个 utf-8 文本文件中获取文本,并按 block 执行以提高性能。
std::ifstream.read(myChunkBuff_str, myChunkBuff_str.length())
Here is a more detailed example
我在每个 block 中得到大约 16,000 个字符。我的下一步是将这个 std::string
转换成可以让我单独处理这些“复杂字符”的东西,从而将那个 std::string
转换成 std::wstring
。
我正在使用以下函数进行转换,taken from here:
#include <string>
#include <codecvt>
#include <locale>
std::string narrow (const std::wstring& wide_string)
{
std::wstring_convert <std::codecvt_utf8 <wchar_t>, wchar_t> convert;
return convert.to_bytes (wide_string);
}
std::wstring widen (const std::string& utf8_string)
{
std::wstring_convert <std::codecvt_utf8 <wchar_t>, wchar_t> convert;
return convert.from_bytes (utf8_string);
}
但是,在 block 的末尾,其中一个俄语字符可能会被截断,并且转换将失败,并出现 std::range_error 异常
。
例如,在 UTF-8 中,“привет”占用 15 个字符,“приве”占用 13 个字符。因此,如果我的 block 假设为 14,则“т”将部分丢失,并且转换将引发异常。
问题:
如何检测这些部分加载的字符? (在这种情况下为 'т')这将允许我在没有它的情况下进行转换,并且可能比计划提前一点移动下一个 block ,以便下次包含这个有问题的 'т'?
我不想围绕这些函数try
或catch
,因为try/catch 可能会减慢我的程序速度。它也没有告诉我“要使转换真正成功,缺少多少字符”。
我知道 wstring_convert::converted()
但如果我的程序在我开始之前就崩溃了,它就没什么用了
最佳答案
您可以使用几个函数来完成此操作。 UTF-8
有一种方法可以检测多字节字符的开头和(从开头)多字节字符的大小。
所以两个函数:
// returns zero if this is the first byte of a UTF-8 char
// otherwise non-zero.
static unsigned is_continuation(char c)
{
return (c & 0b10000000) && !(c & 0b01000000);
}
// if c is the *first* byte of a UTF-8 multibyte character, returns
// the total number of bytes of the character.
static unsigned size(const unsigned char c)
{
constexpr static const char u8char_size[] =
{
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2
, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2
, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3
, 4, 4, 4, 4, 4, 4, 4, 4, 5, 5, 5, 5, 6, 6, 0, 0
};
return u8char_size[(unsigned char)c];
}
您可以从缓冲区的末尾回溯,直到 is_continuation(c)
为 false。然后检查当前 UTF-8
字符的 size(c)
是否比缓冲区的末尾长。
免责声明 - 上次我看到这些功能可以正常工作,但已经有一段时间没有使用了。
编辑:添加。
如果您想手动完成整个操作,我也可以发布将 UTF-8
多字节字符转换为 UTF-16
多字节或 UTF-32
字符。
UTF-32 很简单:
// returns a UTF-32 char from a `UTF-8` multibyte
// character pointed to by cp
static char32_t char32(const char* cp)
{
auto sz = size(*cp); // function above
if(sz == 1)
return *cp;
char32_t c32 = (0b0111'1111 >> sz) & (*cp);
for(unsigned i = 1; i < sz; ++i)
c32 = (c32 << 6) | (cp[i] & 0b0011'1111);
return c32;
}
UTF-16 有点棘手:
// UTF-16 characters can be 1 or 2 characters wide...
using char16_pair = std::array<char16_t, 2>;
// outputs a UTF-16 char in cp16 from a `UTF-8` multibyte
// character pointed to by cp
//
// returns the number of characters in this `UTF-16` character
// (1 or 2).
static unsigned char16(const char* cp, char16_pair& cp16)
{
char32_t c32 = char32(cp);
if(c32 < 0xD800 || (c32 > 0xDFFF && c32 < 0x10000))
{
cp16[0] = char16_t(c32);
cp16[1] = 0;
return 1;
}
c32 -= 0x010000;
cp16[0] = ((0b1111'1111'1100'0000'0000 & c32) >> 10) + 0xD800;
cp16[1] = ((0b0000'0000'0011'1111'1111 & c32) >> 00) + 0xDC00;
return 2;
}
关于c++ - 'std::wstring_convert' 尽可能多地转换(从 UTF8 文件读取 block ),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52338904/
我们正在使用一个 std::wstring_convert> converter; 在我们的记录器中,它从遗留组件获取 UTF-16 字符串并将其转换为我们写入日志的 UTF-8。转换器在每次转换时都
我希望能够将从文件中读取的文本转换为多字节字符。我在 Windows 上有以下适用于我的 C++ 代码。当我尝试在 Linux 上编译代码时失败了。 #include .... std::wstri
我有 C++ 代码将包含 🜆( ALCHEMICAL SYMBOL FOR AQUA REGIA ) 的 string 转换为 u16string: #include #include #inc
我正在编写一些需要使用系统语言环境在字节字符串和宽字符串之间进行转换的代码。从文件中读取时,这非常容易做到。我可以使用 std::wifstream , 用 std::locale("") 灌输它,
这个问题在这里已经有了答案: Specification of source charset encoding in MSVC++, like gcc "-finput-charset=CharSe
我正在计划使用 C++Builder 10.1 (Clang 3.3) 的新 C++11 Win32/64 项目,并考虑在核心功能方面以最便携的方式实现它,所以我想使用 UTF-8 作为 std::s
我正在使用 std::wstring_convert 将 wstring 转换为多字节字符串,如下所示: // convert from wide char to multibyte char
在头文件 locale 中声明了两个方便的接口(interface):std::wstring_convert 和 std::wbuffer_convert。但是,没有使用示例。 有没有简明的例子说明
我正在从一个 utf-8 文本文件中获取文本,并按 block 执行以提高性能。 std::ifstream.read(myChunkBuff_str, myChunkBuff_str.length(
当您从 std::u16string 转到时,假设 std::u32string,std::wstring_convert 不起作用正如它所期望的那样 chars。那么如何使用 std::wstrin
C++11 的 std::wstring_convert效果很好 * 用于标准 UTF-8 UTF-16/UCS2/UCS4 转换。但是,当我尝试使用不是来自 的构面实例化 wstring_con
我是一名优秀的程序员,十分优秀!