在串口传递数据的时候数据格式是16进制的,但是存下来的数据为了方便查看,用字符串存储下来了。例如串口传递的数据是0xAA 0xBB 0xCC,存储下来的文件为AABBCC,这样存储带来的问题就是无法直接通过串口再发送出去,所以需要转换一下。在实现转换之前,先看一个16进制和字符的对应关系,如下所述。
用16进制打开一个文件如下图:
文件的大小是5个字节
为啥文件里面以01作为一个字节呢。原因是01是两个16进制的字符,一个16进制字符是4位,8位是一个字节,所以01表示一个字节。基于上面的结论,转换函数实现如下:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#define MAX_DATA_SIZE 1024
static int convert_string2hex(unsigned char *in_data, int in_data_len, unsigned char *out_data, int *out_data_len)
{
int i;
int loop_count;
int convert_point = 0;
int mem_point = 0;
unsigned char convert_result;
unsigned char temp[3] = {0}; /* 为啥是3呢? */
/* 检查参数有效性 */
if (in_data == NULL || in_data_len <= 0 || out_data == NULL || out_data_len == NULL || (in_data_len % 2) != 0) {
printf("invalid parameters\n");
return -1;
}
/* 判断是否超过16进制范围 0 ~ F */
for (i = 0; i < in_data_len; i++) {
if ((in_data[i] < '0') || (in_data[i] > 'f') || ((in_data[i] > '9') &&(in_data[i] < 'A'))) {
printf("out of range\n");
return -1;
}
}
loop_count = in_data_len / 2;
memset(out_data, 0x00, *out_data_len);
*out_data_len = 0;
for (i = 0; i < loop_count; i++) {
memset(temp, 0x00, sizeof(temp));
memcpy(temp, in_data + convert_point, 2);
convert_point += 2;
convert_result = strtoul(temp, NULL, 16);
memcpy(out_data + mem_point, &convert_result, sizeof(unsigned char));
mem_point += sizeof(unsigned char);
*out_data_len += sizeof(unsigned char);
}
return 0;
}
测试转换功能的代码如下:
int main(int argc, char const *argv[])
{
unsigned char in_data[MAX_DATA_SIZE] = {0};
int in_data_len = MAX_DATA_SIZE;
unsigned char out_data[MAX_DATA_SIZE] = {0};
int out_data_len = MAX_DATA_SIZE;
int i;
scanf("%s", in_data);
in_data_len = strlen(in_data);
convert_string2hex(in_data, in_data_len, out_data, &out_data_len);
for (i = 0; i < out_data_len; i++) {
printf("0x%02x ", out_data[i]);
}
printf("\n");
return 0;
}
输入AABBCC转换后的结果是0xAA 0xBB 0xCC。
注意点
convert_string2hex()函数中用来承接数据的temp必须要是3个字节,原因是strtoul函数的功能导致,strtoul()函数的定义如下:
unsigned long int strtoul(const char *nptr, char **endptr, int base);
strtoul()会扫描参数nptr字符串,跳过前面的空白字符(例如空格,tab缩进等),直到遇上数字或正负符号才开始做转换,再遇到非数字或字符串结束时(’\0’)结束转换,如果申请的temp是2个字节,那么其后面的一个字节如果不是0,转换就会出错,所以多申请一个字节来防止出现这种错误