平时我们在写功能需求时,难免不会遇到需要将16
进制字符串转换成字符串,或者字符串转16进制字符串的需求,所以就有了这篇文章。
在进入主题之前,我们先来回顾一下字符和16
进制的关系。我们知道正常的16
进制数值(像0x32
)和ASIIC
码(0x32
对应字符2
)之间是互通的,即如果你输出数值,就是输出的字符的数值(例如,字符0
,十进制就是48
),如果你输出字符,那就是输出的字符,不会进行转换(例如,字符0
,输出就是字符0
),具体可以看下面示例:
char ch = '0';int i = 48;printf("int: %d", ch); // 输出: int: 48printf("char: %c", ch); // 输出: char: 0printf("int: %d", i); // 输出: int: 48printf("char: %c", i); // 输出: char: 0
16进制字符串转字符串
言归正传,现在我们先来看看16
进制字符串转字符串,大致的逻辑就是将16
进制字符串每两个字符组合在一起得到一个16
进制的字符串,然后再打包转成对应的字符。我们可以利用库函数strtol()
,其函数声明为long int strtol(const char *str, char **endptr, int base)
,该函数的功能是把参数 str
所指向的字符串根据给定的 base
转换为一个长整数(类型为 long int
型),base
必须介于 2
和 36
(包含)之间,或者是特殊值 0
。
str
– 要转换为长整数的字符串endptr
– 对类型为char*
的对象的引用,其值由函数设置为str
中数值后的下一个字符base
– 基数,必须介于2
和36
(包含)之间,或者特殊值0
。如果base
为0
,则会根据字符串的前缀来判断进制 返回值:函数返回被转换的长整型整数值。如果输入字符串不符合数字格式,
strtol()
将返回 0
。如果转换结果超出了 long
整数的表示范围,那么将产生溢出,并设置 errno
为 ERANGE
。你可以使用 <errno.h>
头文件中的 errno
变量来检查是否有溢出发生。 现在我们来进行实战操作,如下示例:
#include <stdlib.h> // 要使用strtol()库函数,需要包含头文件char data[] = "48656C6C6F20576F726C6421210D0A"; // 假如,我们接收到这样的数据char res[32]; // 储存转换后的结果int hex_str_2_str(char *dest, char *src){ int len = strlen(src); // 获取接收数据长度 int i,j; for (i = 0, j = 0; i < len; i+=2) { // 每次取两个字符 char tmp_buf[3]; // 每两个字符组成一个16进制字符串,同时结尾需要空字符来告诉编译器我们的是字符串 char *endptr; // 保存已转换数值后的下一个字符 // 以下为取待转换的16进制字符串 tmp_buf[0] = src[i]; tmp_buf[1] = src[i + 1]; tmp_buf[2] = '\0'; // 记得添加空字符 // 转换成16进制,base传16即可 dest[j++] = strtol(tmp_buf, &endptr, 16); } dest[j] = '\0'; // 添加字符串结束符 return j;}int len = hex_str_2_str(res, data);printf("len: %d, str: %s", len, res); // len: 15, str: Hello World!!
字符串转16进制字符串
前面介绍了16
进制字符串转字符串,那现在我们来试试如何反过来转换。具体操作是依次将每个字符拿到并转换成对应的16
进制,然后再以字符串的形式储存在容器中即可。
在这里我们需要用到库函数sprintf()
,其函数声明为int sprintf(char *str, const char *format, ...)
,该函数是把格式化输出内容发送到所指向的字符串str
中。
参数:
str
– 指向一个字符数组的指针,该数组存储了C
字符串format
– 这是字符串,包含了要被写入到字符串str
的文本。它可以包含嵌入的format
标签,format
标签可被随后的附加参数中指定的值替换,并按需求进行格式化 返回值:
如果成功,则返回写入的字符总数,不包括字符串追加在字符串末尾的空字符。如果失败,则返回一个负数。
具体操作如下示例:
#include "stdio.h" // sprintf()函数声明所在的头文件char data[] = "Hello World!!\r\n";char res[32];int str_2_hex_str(char *dest, char *src){ int len = strlen(src); // 获取接收数据长度 int i,j; for (i = 0, j = 0; i < len; i++) { // %02X: 是格式化字符串,意思是以大写的形式(通过X大小写控制)转换成16进制,长度不足2的自动补0 sprintf(&dest[j], "%02X", src[i]); j+=2; // 每个16进制占2个长度 } dest[j] = '\0'; // 添加字符串结束符 return j; // 返回字符串长度}int len = str_2_hex_str(res, data);printf("len: %d, hex: %s", len, res); // len: 30, hex: 48656C6C6F20576F726C6421210D0A
总结
要想实现16
进制字符串和字符串之间的相互转换,只要用好strtol()
和sprintf()
这两个库函数,就可以轻松解决
参考
https://www.runoob.com/cprogramming/c-function-strtol.html
https://www.runoob.com/cprogramming/c-function-sprintf.html
关注公众号《嵌入式从入门到放弃》了解更多知识