如果unicode使用17位代码点,那么如何从代码点计算代理对?
答案 0 :(得分:7)
Unicode代码点是标量值,范围从0x000000到0x10FFFF。因此它们是21位整数,而不是17位。
代理对是UTF-16形式的机制。这表示21位标量值为一个或两个16位代码单元。
使用示例代码在Unicode联盟的常见问题解答UTF-8, UTF-16, UTF-32 & BOM中详细解释了这一点。该FAQ引用了Unicode标准中更详细的部分。
答案 1 :(得分:4)
如果是您所使用的代码,以下是单个代码点分别以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));
}
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);
}