字符串和16进制之间的转换

在串口传递数据的时候数据格式是16进制的,但是存下来的数据为了方便查看,用字符串存储下来了。例如串口传递的数据是0xAA 0xBB 0xCC,存储下来的文件为AABBCC,这样存储带来的问题就是无法直接通过串口再发送出去,所以需要转换一下。在实现转换之前,先看一个16进制和字符的对应关系,如下所述。

用16进制打开一个文件如下图:
CSDN
文件的大小是5个字节
CSDN
为啥文件里面以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,转换就会出错,所以多申请一个字节来防止出现这种错误

发布了115 篇原创文章 · 获赞 11 · 访问量 3万+

猜你喜欢

转载自blog.csdn.net/qq_33242956/article/details/93199564