一、字符编码简单介绍

1. ASCII码

在计算机内部,全部的信息终于都表示为一个二进制的字符串。每个二进制位(bit)有0和1两种状态,因此八个二进制位就能够组合出256种状态,这被称为一个字节(byte)。也就是说,一个字节一共能够用来表示256种不同的状态,每个状态相应一个符号,就是256个符号,从0000000到11111111。
上个世纪60年代,美国制定了一套字符编码,对英语字符与二进制位之间的关系,做了统一规定。这被称为ASCII码,一直沿用至今。
ASCII码一共规定了128个字符的编码,比方空格“SPACE”是32(二进制00100000),大写的字母A是65(二进制01000001)。这128个符号(包含32个不能打印出来的控制符号),仅仅占用了一个字节的后面7位,最前面的1位统一规定为0。

2. Unicode

英语用128个符号编码就够了,可是用来表示其它语言,128个符号是不够的。因此,非常多欧洲国家发明了非常多非ASCII码,相同用一个字节,用最高位为1的区间(既128~255)来扩展原来的ASCII码,当中一种比較有名的就是IBM字符编码。这样一来,这些欧洲国家使用的编码体系,能够表示最多256个符号。可是,这里又出现了新的问题。不同的国家有不同的字母,因此,哪怕它们都使用256个符号的编码方式,代表的字母却不一样。比方,130在法语编码中代表了é,在希伯来语编码中却代表了字母Gimel (ג),在俄语编码中又会代表还有一个符号。可是无论如何,全部这些编码方式中,0—127表示的符号是一样的,不一样的仅仅是128—255的这一段。

至于亚洲国家的文字,使用的符号就很多其它了,汉字就多达10万左右。一个字节仅仅能表示256种符号,肯定是不够的,就必须使用多个字节表达一个符号。比方,中文简体常见的编码方式是GB2312,使用两个字节表示一个汉字,所以理论上最多能够表示256x256=65536个符号。

世界上存在着多种编码方式,同一个二进制数字能够被解释成不同的符号。因此,要想打开一个文本文件,就必须知道它的编码方式,否则用错误的编码方式解读,就会出现乱码。为什么电子邮件经常出现乱码?就是由于发信人和收信人使用的编码方式不一样。
能够想象,假设有一种编码,将世界上全部的符号都纳入当中。每个符号都给予一个独一无二的编码,那么乱码问题就会消失。这就是Unicode,就像它的名字都表示的,这是一种全部符号的编码。

Unicode当然是一个非常大的集合,如今的规模能够容纳100多万个符号。每一个符号的编码都不一样。须要注意的是,Unicode仅仅是一个符号集,它仅仅规定了符号的二进制代码,却没有规定这个二进制代码应该怎样存储。

3. UTF-8

互联网的普及,强烈要求出现一种统一的编码方式。UTF-8就是在互联网上使用最广的一种unicode的实现方式。其它实现方式还包含UTF-16和UTF-32,只是在互联网上基本不用。反复一遍,这里的关系是,UTF-8是Unicode的实现方式之中的一个。
UTF-8最大的一个特点,就是它是一种变长的编码方式。它能够使用1~6个字节表示一个符号,依据不同的符号而变化字节长度。
UTF-8的编码规则非常easy,仅仅有二条:
1)对于单字节的符号,字节的第一位设为0,后面7位为这个符号的unicode码。因此对于英语字母,UTF-8编码和ASCII码是同样的。
2)对于n字节的符号(n>1),第一个字节的前n位都设为1,第n+1位设为0,后面字节的前两位一律设为10。剩下的没有提及的二进制位,所有为这个符号的unicode码。
如表: 
1字节 0xxxxxxx 
2字节 110xxxxx 10xxxxxx 
3字节 1110xxxx 10xxxxxx 10xxxxxx 
4字节 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx 
5字节 111110xx 10xxxxxx 10xxxxxx 10xxxxxx 10xxxxxx 
6字节 1111110x 10xxxxxx 10xxxxxx 10xxxxxx 10xxxxxx 10xxxxxx 
因此UTF-8中能够用来表示字符编码的实际位数最多有31位,即上表中x所表示的位。除去那些控制位(每字节开头的10等),这些x表示的位与UNICODE编码是一一相应的,位高低顺序也同样。 
实际将UNICODE转换为UTF-8编码时应先去除高位0,然后依据所剩编码的位数决定所需最小的UTF-8编码位数。 
因此那些基本ASCII字符集中的字符(UNICODE兼容ASCII)仅仅须要一个字节的UTF-8编码(7个二进制位)便能够表示。

依据此规则,能够非常方便的把UTF-8编码的字符串拆分成单字集合,代码例如以下:

  size_t utf8_to_charset(const std::string &input, std::set<std::string> &output) {std::string ch; for (size_t i = 0, len = 0; i != input.length(); i += len) {unsigned char byte = (unsigned)input[i];if (byte >= 0xFC) // lenght 6len = 6;  else if (byte >= 0xF8)len = 5;else if (byte >= 0xF0)len = 4;else if (byte >= 0xE0)len = 3;else if (byte >= 0xC0)len = 2;elselen = 1;ch = input.substr(i, len);output.insert(ch);}   return output.size();}

这里我把字符串转换为单字的集合(set)是由于应用场景的须要,假设须要保持单字在字符串中的位置,能够非常方便的用vector来替换set。

以下是获取UTF-8字符串的字符个数(注意,不是字符串长度哦)的代码:

  size_t get_utf8_length(const std::string &input) {size_t length = 0;for (size_t i = 0, len = 0; i != input.length(); i += len) {unsigned char byte = input[i];if (byte >= 0xFC) // lenght 6len = 6;  else if (byte >= 0xF8)len = 5;else if (byte >= 0xF0)len = 4;else if (byte >= 0xE0)len = 3;else if (byte >= 0xC0)len = 2;elselen = 1;length ++;}   return length;}     

转载于:https://www.cnblogs.com/mengfanrong/p/3785791.html

UTF-8编码的字符串拆分成单字、获取UTF-8字符串的字符个数的代码及原理(c++实现)...相关推荐

  1. Java 将字符串拆分成数组,实现字符串组合

    将字符串拆分成数组和实现字符串组合 不能使用语言的基本分割组合函数(如 Java 的 String.split, php 的 explode 和 implode) 1) 字符串拆分成字符串数组,如&q ...

  2. 把一个字符串分割成数组 php_PHP怎么将字符串拆分成数组

    在日常项目开发过程中,较长的字符串可能需要被拆分成数组形式,以便被展现或用于判断验证.那么将字符串拆分成数组,也很容易实现,我们可以直接通过PHP中的explode函数来进行拆分. 下面我们就通过简单 ...

  3. vue把字符串分割成等长的若干字符串,根据特定字符分割字符串

    一.把字符串分割成等长的若干字符串 1. 把字符串分割成特定长度的若干字符串(数组形式),这里举一个每5个字符为一组的例子 const strArr = 'new计算-架构&编程-语言'.ma ...

  4. 每天一道LeetCode-----将字符串拆分成有效的ip地址

    Restore IP Addresses 原题链接Restore IP Addresses 给定一个仅包含数字的字符串,将其拆分成有效的ip地址 题目的主要意思实际是在字符串中添加三个点,从而构成一个 ...

  5. python将字符串拆分成单词_将字符串拆分为单词和标点符号

    我正在尝试将字符串拆分为单词和标点符号,并将标点符号添加到拆分生成的列表中. 例如: >>> c = "help, me" >>> print ...

  6. python把一个字符串拆分成多个字段_如何把一个字符串分割成多个字符串

    展开全部 下面两种方法均可以把一个62616964757a686964616fe58685e5aeb931333363393661字符串分割成多个字符串 一.Java 知识补充String的split ...

  7. mybatis字符串转成数字_计算机储存数字和字符的方法你了解多少?

    前语:不要为了读文章而读文章,一定要带着问题来读文章,勤思考. 本文链接: http://1t.click/J7E 前言 最近在学习中涉及到计算机储存.传输数字和字符等操作,由于对字节.2进制.10进 ...

  8. c字符串分割成数组_数组与字符串

    定义数组时,应该注意以下几点: (1) 数组使用的是方括号[ ],不要误写成小括号( ). (2) 常量表达式的值必须是一个正整数值,不能含有变量,但是可以是符号常数或常量表达式. (3) 数组定义后 ...

  9. c语言字符串转成二进制,C语言中字符串如何转换为二进制、八进制、十进制、十六进制...

    在C语言某个程序当中需要把文本16进制转换成对应的16进制数,比如字符串"0x1a"转换成10进制的26,可以用以下函数来实现 相关函数: atof, atoi, atol, st ...

最新文章

  1. ThinkPHP入门(二)
  2. setTimeout 定时器的使用
  3. 面对大规模系统工程,看Facebook如何处理故障排查(一)
  4. 正三角形旋转一周得到的图形是_七年级数学上册_第一章《基本的几何图形》_知识点...
  5. abaqus python 建立节点集合_在Python中创建Abaqus集
  6. C++设计模式-模板方法模式
  7. HashMap 排序
  8. Numpy的使用方法
  9. 白鹭引擎 - 矢量绘图 ( graphics )
  10. Thread类的常用方法
  11. iOS字体包导入过程
  12. 《python学习手册》目录
  13. 不错的Unity教程
  14. 分析Python7个爬虫小案例(附源码)
  15. ar机房查看html5,探秘5G“智慧机房”:5G+AR技术实现机房智能巡检
  16. 互联网+时代的到来,让一站式婚庆管理系统成为潮流
  17. Feature Tools:自动特征工程(翻译)
  18. edup无线网卡驱动安装linux,EDUP EP-N8513 (RTL8188CUS芯片)在Ubuntu 12.10下的wifi不能连接问题解决方法...
  19. iOS编译cURL库并链接darwinssl,zlib,c-ares库操作步骤
  20. Python基础——继承、多态

热门文章

  1. 一文看懂Python(五)-----文件篇
  2. STM32开发 -- 外部中断详解
  3. 记录一次socket编程:String的trim函数
  4. 高一计算机专业用平板电脑,高中生需不需要平板电脑或者笔记本电脑?
  5. oracle 分析视图,ORACLE 性能视图的分析和使用
  6. Parity 錢包合約漏洞
  7. Generative Adversarial Networks 生成对抗网络的简单理解
  8. JZOJ 5474. 【NOIP2017提高组正式赛】时间复杂度
  9. POJ 3580. SuperMemo
  10. python哪个方向工资高_Python薪资最高的是一线城市,竟然平均月薪高达20k!