C语言16进制字符串转数字

Love The Way You Lie 2022-08-28 13:50 300阅读 0赞

假设有一个十六进制字符串,“99AD1B5226A37E3E058E3B8E27C2C666”,如何把它按照字节来切分,然后转成整型?也就是说转换成 0x99, 0xAD, 0x1B, …

代码实现

C 语言参考代码:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. char *key = "99AD1B5226A37E3E058E3B8E27C2C666";
  5. void main(void)
  6. {
  7. if(strlen(key) & 1){
  8. printf("key 的长度是奇数!");
  9. return;
  10. }
  11. char byte[3] = { 0};
  12. char *p = key;
  13. char *endptr;
  14. unsigned int res;
  15. for(int i = 0; i < strlen(key); i += 2){
  16. memcpy(byte, &key[i], 2);
  17. res = strtol(byte, &endptr, 16);
  18. printf("%02x ", res);
  19. }
  20. printf("\n");
  21. }

测试结果:

  1. 99 ad 1b 52 26 a3 7e 3e 05 8e 3b 8e 27 c2 c6 66

说明:

  1. 代码中要求待转换字符串的长度是偶数,如果是奇数,可以手动补零
  2. 用到的关键函数是 strtol,它的原型是

    long int strtol (const char str, char* endptr, int base);

strtol 介绍

  1. long int strtol (const char* str, char** endptr, int base);

【参数说明】
str 为要转换的字符串,endstr 为第一个不能转换的字符的指针,base 为字符串 str 所采用的进制。
【函数说明】
strtol() 会扫描参数 str 字符串,跳过前面的空白字符(例如空格,tab缩进等,可以通过 isspace() 函数来检测),直到遇上数字或正负符号才开始做转换,再遇到非数字或字符串结束时(’\0’)结束转换,并将结果返回。

两点注意:

  1. 当 base 的值为 0 时,默认采用 10 进制转换,但如果遇到 ‘0x’ / ‘0X’ 前置字符则会使用 16 进制转换,遇到 ‘0’ 前置字符则会使用 8 进制转换。
  2. 若endptr 不为NULL,则会将遇到的不符合条件而终止的字符指针由 endptr 传回;若 endptr 为 NULL,则表示该参数无效,或不使用该参数。

【返回值】
返回转换后的长整型数;如果不能转换或者 str 为空字符串,那么返回 0(0L);如果转换得到的值超出 long int 所能表示的范围,函数将返回 LONG_MAX 或 LONG_MIN(在 limits.h 头文件中定义),并将 errno 的值设置为 ERANGE。

代码重构

为了方便使用,可以把这个小功能提炼成一个函数,我给它起名为 hex_string_to_u8。

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. typedef unsigned char u8;
  5. int hex_string_to_u8(const char *hex_str, char *out)
  6. {
  7. if(strlen(hex_str) & 1){
  8. printf("字符串的长度是奇数!");
  9. return -1;
  10. }
  11. char byte[3] = { 0};
  12. const char *p = hex_str;
  13. int j = 0;
  14. for(int i=0; i<strlen(hex_str); i+=2){
  15. memcpy(byte, &p[i], 2);
  16. out[j++] = strtol(byte, NULL, 16);
  17. }
  18. return 0;
  19. }
  20. void hexdump(void *data, int size){
  21. if (size <= 0) return;
  22. int i;
  23. for (i=0; i<size;i++){
  24. printf("%02X ", ((u8 *)data)[i]);
  25. }
  26. printf("\n");
  27. }
  28. void main(void)
  29. {
  30. char *key = "99AD1B5226A37E3E058E3B8E27C2C666";
  31. char res[16] = { 0};
  32. hex_string_to_u8(key, res);
  33. hexdump(res, 16);
  34. }

参考资料:
C语言strtol()函数:将字符串转换成long(长整型数)_C语言中文网

发表评论

表情:
评论列表 (有 0 条评论,300人围观)

还没有评论,来说两句吧...

相关阅读