简介

1、这段代码只考虑在小端序情况下的转换(一般的机器都是的)。

2、这段代码需要C++11的支持(只是用到了u16string),如果不支持,可以添加下面代码

typedef uint16_t char16_t;
typedef std::basic_string<char16_t>

utfconvert.h

#ifndef __UTFCONVERT_H__
#define __UTFCONVERT_H__
#include <string> // 从UTF16编码字符串构建,需要带BOM标记
std::string utf16_to_utf8(const std::u16string& u16str); // 从UTF16 LE编码的字符串创建
std::string utf16le_to_utf8(const std::u16string& u16str); // 从UTF16BE编码字符串创建
std::string utf16be_to_utf8(const std::u16string& u16str); // 获取转换为UTF-16 LE编码的字符串
std::u16string utf8_to_utf16le(const std::string& u8str, bool addbom = false, bool* ok = NULL); // 获取转换为UTF-16 BE的字符串
std::u16string utf8_to_utf16be(const std::string& u8str, bool addbom = false, bool* ok = NULL); #endif //! __UTFCONVERT_H__

utfconvert.cpp

#include "utfconvert.h"

#include <stdint.h>
#ifdef __GNUC__
#include <endian.h>
#endif // __GNUC__ static inline uint16_t byteswap_ushort(uint16_t number)
{
#if defined(_MSC_VER) && _MSC_VER > 1310
return _byteswap_ushort(number);
#elif defined(__GNUC__)
return __builtin_bswap16(number);
#else
return (number >> 8) | (number << 8);
#endif
} ////////////////////////////////////////
// 以下转换都是在小端序下进行 //
//////////////////////////////////////// // 从UTF16编码字符串构建,需要带BOM标记
std::string utf16_to_utf8(const std::u16string& u16str)
{
if (u16str.empty()){ return std::string(); }
//Byte Order Mark
char16_t bom = u16str[0];
switch (bom){
case 0xFEFF: //Little Endian
return utf16le_to_utf8(u16str);
break;
case 0xFFFE: //Big Endian
return utf16be_to_utf8(u16str);
break;
default:
return std::string();
}
} // 从UTF16 LE编码的字符串创建
std::string utf16le_to_utf8(const std::u16string& u16str)
{
if (u16str.empty()){ return std::string(); }
const char16_t* p = u16str.data();
std::u16string::size_type len = u16str.length();
if (p[0] == 0xFEFF){
p += 1; //带有bom标记,后移
len -= 1;
} // 开始转换
std::string u8str;
u8str.reserve(len * 3); char16_t u16char;
for (std::u16string::size_type i = 0; i < len; ++i){
// 这里假设是在小端序下(大端序不适用)
u16char = p[i]; // 1字节表示部分
if (u16char < 0x0080){
// u16char <= 0x007f
// U- 0000 0000 ~ 0000 07ff : 0xxx xxxx
u8str.push_back((char)(u16char & 0x00FF)); // 取低8bit
continue;
}
// 2 字节能表示部分
if (u16char >= 0x0080 && u16char <= 0x07FF){
// * U-00000080 - U-000007FF: 110xxxxx 10xxxxxx
u8str.push_back((char)(((u16char >> 6) & 0x1F) | 0xC0));
u8str.push_back((char)((u16char & 0x3F) | 0x80));
continue;
}
// 代理项对部分(4字节表示)
if (u16char >= 0xD800 && u16char <= 0xDBFF) {
// * U-00010000 - U-001FFFFF: 1111 0xxx 10xxxxxx 10xxxxxx 10xxxxxx
uint32_t highSur = u16char;
uint32_t lowSur = p[++i];
// 从代理项对到UNICODE代码点转换
// 1、从高代理项减去0xD800,获取有效10bit
// 2、从低代理项减去0xDC00,获取有效10bit
// 3、加上0x10000,获取UNICODE代码点值
uint32_t codePoint = highSur - 0xD800;
codePoint <<= 10;
codePoint |= lowSur - 0xDC00;
codePoint += 0x10000;
// 转为4字节UTF8编码表示
u8str.push_back((char)((codePoint >> 18) | 0xF0));
u8str.push_back((char)(((codePoint >> 12) & 0x3F) | 0x80));
u8str.push_back((char)(((codePoint >> 06) & 0x3F) | 0x80));
u8str.push_back((char)((codePoint & 0x3F) | 0x80));
continue;
}
// 3 字节表示部分
{
// * U-0000E000 - U-0000FFFF: 1110xxxx 10xxxxxx 10xxxxxx
u8str.push_back((char)(((u16char >> 12) & 0x0F) | 0xE0));
u8str.push_back((char)(((u16char >> 6) & 0x3F) | 0x80));
u8str.push_back((char)((u16char & 0x3F) | 0x80));
continue;
}
} return u8str;
} // 从UTF16BE编码字符串创建
std::string utf16be_to_utf8(const std::u16string& u16str)
{
if (u16str.empty()){ return std::string(); }
const char16_t* p = u16str.data();
std::u16string::size_type len = u16str.length();
if (p[0] == 0xFEFF){
p += 1; //带有bom标记,后移
len -= 1;
} // 开始转换
std::string u8str;
u8str.reserve(len * 2);
char16_t u16char; //u16le 低字节存低位,高字节存高位
for (std::u16string::size_type i = 0; i < len; ++i) {
// 这里假设是在小端序下(大端序不适用)
u16char = p[i];
// 将大端序转为小端序
u16char = byteswap_ushort(u16char); // 1字节表示部分
if (u16char < 0x0080) {
// u16char <= 0x007f
// U- 0000 0000 ~ 0000 07ff : 0xxx xxxx
u8str.push_back((char)(u16char & 0x00FF));
continue;
}
// 2 字节能表示部分
if (u16char >= 0x0080 && u16char <= 0x07FF) {
// * U-00000080 - U-000007FF: 110xxxxx 10xxxxxx
u8str.push_back((char)(((u16char >> 6) & 0x1F) | 0xC0));
u8str.push_back((char)((u16char & 0x3F) | 0x80));
continue;
}
// 代理项对部分(4字节表示)
if (u16char >= 0xD800 && u16char <= 0xDBFF) {
// * U-00010000 - U-001FFFFF: 1111 0xxx 10xxxxxx 10xxxxxx 10xxxxxx
uint32_t highSur = u16char;
uint32_t lowSur = byteswap_ushort(p[++i]);
// 从代理项对到UNICODE代码点转换
// 1、从高代理项减去0xD800,获取有效10bit
// 2、从低代理项减去0xDC00,获取有效10bit
// 3、加上0x10000,获取UNICODE代码点值
uint32_t codePoint = highSur - 0xD800;
codePoint <<= 10;
codePoint |= lowSur - 0xDC00;
codePoint += 0x10000;
// 转为4字节UTF8编码表示
u8str.push_back((char)((codePoint >> 18) | 0xF0));
u8str.push_back((char)(((codePoint >> 12) & 0x3F) | 0x80));
u8str.push_back((char)(((codePoint >> 06) & 0x3F) | 0x80));
u8str.push_back((char)((codePoint & 0x3F) | 0x80));
continue;
}
// 3 字节表示部分
{
// * U-0000E000 - U-0000FFFF: 1110xxxx 10xxxxxx 10xxxxxx
u8str.push_back((char)(((u16char >> 12) & 0x0F) | 0xE0));
u8str.push_back((char)(((u16char >> 6) & 0x3F) | 0x80));
u8str.push_back((char)((u16char & 0x3F) | 0x80));
continue;
}
}
return u8str;
} // 获取转换为UTF-16 LE编码的字符串
std::u16string utf8_to_utf16le(const std::string& u8str, bool addbom, bool* ok)
{
std::u16string u16str;
u16str.reserve(u8str.size());
if (addbom) {
u16str.push_back(0xFEFF); //bom (字节表示为 FF FE)
}
std::string::size_type len = u8str.length(); const unsigned char* p = (unsigned char*)(u8str.data());
// 判断是否具有BOM(判断长度小于3字节的情况)
if (len > 3 && p[0] == 0xEF && p[1] == 0xBB && p[2] == 0xBF){
p += 3;
len -= 3;
} bool is_ok = true;
// 开始转换
for (std::string::size_type i = 0; i < len; ++i) {
uint32_t ch = p[i]; // 取出UTF8序列首字节
if ((ch & 0x80) == 0) {
// 最高位为0,只有1字节表示UNICODE代码点
u16str.push_back((char16_t)ch);
continue;
}
switch (ch & 0xF0)
{
case 0xF0: // 4 字节字符, 0x10000 到 0x10FFFF
{
uint32_t c2 = p[++i];
uint32_t c3 = p[++i];
uint32_t c4 = p[++i];
// 计算UNICODE代码点值(第一个字节取低3bit,其余取6bit)
uint32_t codePoint = ((ch & 0x07U) << 18) | ((c2 & 0x3FU) << 12) | ((c3 & 0x3FU) << 6) | (c4 & 0x3FU);
if (codePoint >= 0x10000)
{
// 在UTF-16中 U+10000 到 U+10FFFF 用两个16bit单元表示, 代理项对.
// 1、将代码点减去0x10000(得到长度为20bit的值)
// 2、high 代理项 是将那20bit中的高10bit加上0xD800(110110 00 00000000)
// 3、low 代理项 是将那20bit中的低10bit加上0xDC00(110111 00 00000000)
codePoint -= 0x10000;
u16str.push_back((char16_t)((codePoint >> 10) | 0xD800U));
u16str.push_back((char16_t)((codePoint & 0x03FFU) | 0xDC00U));
}
else
{
// 在UTF-16中 U+0000 到 U+D7FF 以及 U+E000 到 U+FFFF 与Unicode代码点值相同.
// U+D800 到 U+DFFF 是无效字符, 为了简单起见,这里假设它不存在(如果有则不编码)
u16str.push_back((char16_t)codePoint);
}
}
break;
case 0xE0: // 3 字节字符, 0x800 到 0xFFFF
{
uint32_t c2 = p[++i];
uint32_t c3 = p[++i];
// 计算UNICODE代码点值(第一个字节取低4bit,其余取6bit)
uint32_t codePoint = ((ch & 0x0FU) << 12) | ((c2 & 0x3FU) << 6) | (c3 & 0x3FU);
u16str.push_back((char16_t)codePoint);
}
break;
case 0xD0: // 2 字节字符, 0x80 到 0x7FF
case 0xC0:
{
uint32_t c2 = p[++i];
// 计算UNICODE代码点值(第一个字节取低5bit,其余取6bit)
uint32_t codePoint = ((ch & 0x1FU) << 12) | ((c2 & 0x3FU) << 6);
u16str.push_back((char16_t)codePoint);
}
break;
default: // 单字节部分(前面已经处理,所以不应该进来)
is_ok = false;
break;
}
}
if (ok != NULL) { *ok = is_ok; } return u16str;
} // 获取转换为UTF-16 BE的字符串
std::u16string utf8_to_utf16be(const std::string& u8str, bool addbom, bool* ok)
{
// 先获取utf16le编码字符串
std::u16string u16str = utf8_to_utf16le(u8str, addbom, ok);
// 将小端序转换为大端序
for (size_t i = 0; i < u16str.size(); ++i) {
u16str[i] = byteswap_ushort(u16str[i]);
}
return u16str;
}

C++ UTF8和UTF16互转代码的更多相关文章

  1. JS中UTF-8和UTF-16互转

    1.由于服务端使用的Go,默认是使用UTF-8编码的,而JS默认是Unicode编码的(也就是UTF-16),所以为了字符串编码的一致性,将前端字符串数据编码转换为UTF-8之后再发送给服务端,服务端 ...

  2. UTF8 与 UTF16 编码

    Unicode 的发展,英文好的直接去 unicode.org 上去看吧,不好的可以移步到这里 看dengyunze的总结:<关于UTF8,UTF16,UTF32,UTF16-LE,UTF16- ...

  3. UTF-8、UTF-16、UTF-32编码的相互转换

    最近在考虑写一个可以跨平台的通用字符串类,首先需要搞定的就是编码转换问题. vs默认保存代码文件,使用的是本地code(中文即GBK,日文即Shift-JIS),也可以使用带BOM的UTF-8.gcc ...

  4. UTF-8、UTF-16、UTF-32编码的相互转换(不使用现成的函数)

    最近在考虑写一个可以跨平台的通用字符串类,首先需要搞定的就是编码转换问题. vs默认保存代码文件,使用的是本地code(中文即GBK,日文即Shift-JIS),也可以使用带BOM的UTF-8.gcc ...

  5. 简单几句话总结Unicode,UTF-8和UTF-16

    概念 先说一说基本的概念,这包括什么是Unicode,什么是UTF-8,什么是UTF-16. Unicode,UTF-8,UTF-16完整的说明请参考Wiki(Unicode,UTF-8,UTF-16 ...

  6. Unicode、UTF-8、UTF-16和UTF-32的区别

    Unicode是一个巨大的字符集,给世界上所有的字符定义了一个唯一编码.其仅仅规定了每个符号的二进制代码,没有制定细化的存储规则.UTF-8.UTF-16.UTF-32才是Unicode的存储格式定义 ...

  7. 字符编码的种类:ASCII、GB2312、GBK、GB18030、Unicode、UTF-8、UTF-16、Base64

    ASCII码ASCII:https://zh.wikipedia.org/wiki/ASCIIASCII(American Standard Code for Information Intercha ...

  8. 聊聊计算机中的编码(Unicode,GBK,ASCII,utf8,utf16,ISO8859-1等)以及乱码问题的解决办法

    作为一个程序员,一个中国的程序员,想来“乱码”问题基本上都遇到过,也为之头疼过.出现乱码问题的根本原因是编码与解码使用了不同而且不兼容的“标准”,在国内一般出现在中文的编解码过程中. 我们平时常见的编 ...

  9. 字符编码的故事:ASCII,GB2312,Unicode,UTF-8,UTF-16

    http://blog.csdn.net/longintchar/article/details/51079340 ****************************************** ...

随机推荐

  1. [转]mysql在已有无分区表增加分区,mysql5.5才有,可以是innodb_file_per_table关闭状态.

    FROM : http://blog.csdn.net/sunvince/article/details/7752662 mysql5.1的时候新增的partition,解决了比较简单的shardin ...

  2. 未能加载文件或程序集“Newtonsoft.Json”或它的某一个依赖项。找到的程序集清单定义与程序集引用不匹配。 (异常来自 HRESULT:0x80131040)

    "/"应用程序中的服务器错误. 未能加载文件或程序集"Newtonsoft.Json"或它的某一个依赖项.找到的程序集清单定义与程序集引用不匹配. (异常来自 ...

  3. Go语言之进阶篇Socket编程

    一.Socket编程 1.什么是Socket Socket起源于Unix,而Unix基本哲学之一就是“一切皆文件”,都可以用“打开open –> 读写write/read –> 关闭clo ...

  4. HDU—— 5159 Building Blocks

    Problem Description After enjoying the movie,LeLe went home alone. LeLe decided to build blocks. LeL ...

  5. vRealize 7中部署blueprint如果失败,如何不让已经被provision的虚机被删掉?

    正解为: _debug_deployment 哪里用: 参考资料 ========= Custom Properties for Naming and Analyzing Deployments ht ...

  6. 如何让Domain里的其他系统通过DC来进行外网的DNS解析

    搭建一个测试环境, 一般会建立一个DC, 然后再建立许多虚机加入到这个新DC的domain. 我们有个DNS服务器的地址, 哪台虚机要上外网, 就把这个DNS地址填到这台虚机的DNS server a ...

  7. Mac-OSX下Ruby更新

    Mac下是自带Ruby环境的,在有些情况我们是需要更新Ruby的,安装和更新Ruby环境可以通过rvm命令进行操作,rvm在安装过程中通过HomeBrew安装依赖包,如果之前没有装过HomeBrew, ...

  8. JQuery效率问题

    1,前言 我们开发了一个专题系统,生成了JSON的数据格式,采用JQuery动态插入HTML中,在前期的使用中,没有太大的问题,效率还可以接受,但是最近可能由于网络加之页面设计问题,我们的JS效率比较 ...

  9. Android实现录音的方法(最重要的是对MediaRecorder的试用方法)

    package cn.eoe.record; import java.io.File; import java.io.IOException; import android.app.Activity; ...

  10. Android模仿三星手机系统滑动条滑动时滑块变大的特效

    使用三星手机的过程中发现三星手机系统自带的滑动条有一个特效.比方调节亮度的滑动条.在滑动滑块的过程中,滑块会变大.功能非常小可是体验却非常好,于是决定做一个这种效果出来.好了废话不多说了,以下開始实现 ...