Unicode代码点是标量值,范围从0x000000到0x10FFFF.因此它们是21位整数,而不是17位.
代理对是UTF-16形式的机制.这表示21位标量值为一个或两个16位代码单元.
在Unicode联盟的FAQ,UTF-8,UTF-16,UTF-32和BOM中详细解释了这个示例代码.该FAQ引用了Unicode标准中更详细的部分.
如果您要使用的是代码,请按照以下方法分别在UTF-16和UTF-8中编码单个代码点。
UTF-16代码单元的单个代码点:
if (cp < 0x10000u)
{
*out++ = static_cast<uint16_t>(cp);
}
else
{
*out++ = static_cast<uint16_t>(0xd800u + (((cp - 0x10000u) >> 10) & 0x3ffu));
*out++ = static_cast<uint16_t>(0xdc00u + ((cp - 0x10000u) & 0x3ffu));
}
Run Code Online (Sandbox Code Playgroud)
UTF-8码元的单个码点:
if (cp < 0x80u)
{
*out++ = static_cast<uint8_t>(cp);
}
else if (cp < 0x800u)
{
*out++ = static_cast<uint8_t>((cp >> 6) & 0x1fu | 0xc0u);
*out++ = static_cast<uint8_t>((cp & 0x3fu) | 0x80u);
}
else if (cp < 0x10000u)
{
*out++ = static_cast<uint8_t>((cp >> 12) & 0x0fu | 0xe0u);
*out++ = static_cast<uint8_t>(((cp >> 6) & 0x3fu) | 0x80u);
*out++ = static_cast<uint8_t>((cp & 0x3fu) | 0x80u);
}
else
{
*out++ = static_cast<uint8_t>((cp >> 18) & 0x07u | 0xf0u);
*out++ = static_cast<uint8_t>(((cp >> 12) & 0x3fu) | 0x80u);
*out++ = static_cast<uint8_t>(((cp >> 6) & 0x3fu) | 0x80u);
*out++ = static_cast<uint8_t>((cp & 0x3fu) | 0x80u);
}
Run Code Online (Sandbox Code Playgroud)