C语言 十进制整数字符串转十六进制字符串

╰+哭是因爲堅強的太久メ 2023-01-17 07:44 260阅读 0赞

【问题描述】
例如:输入字符串str1[]=“87632394”,需转成字符串str2[]=“5392A0A”

【解决方案】
方案1:
先使用strtol函数将字符串转成十进制数,再通过sprintf函数将十进制数按十六进制字符串形式输出到字符数组保存
该方案弊端:转换的数不能大于0x7FFFFFFF,因为strtol的返回值是long型

  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <stdlib.h>
  4. int main(void)
  5. {
  6. char str1[] = "87632394";
  7. char str2[100] = { 0};
  8. char* endptr;
  9. long data = 0;
  10. data = strtol(str1,&endptr,10);
  11. printf("data=%ld\r\n",data);
  12. sprintf(str2,"%X",data);
  13. printf("str2=%s\r\n",str2);
  14. return 0;
  15. }

运行结果:
在这里插入图片描述
方案2:
解决当要转换的值大于0x7FFFFFFF,如何处理
例如:输入str1[]=”86281102005404”,需转成str2[]=“4E78E26F949C”
首先要解决的是如何把”86281102005404”转成数字保存
借atol函数的理念来自己实现一个

  1. #include <stdio.h>
  2. typedef enum { FALSE = 0, TRUE = !FALSE } BOOL;
  3. __int64 my_atoll(char* str)//__int64可以改成long long,VC++6.0不支持long long,C99中才添加long long
  4. {
  5. __int64 s = 0;
  6. BOOL isMinus = FALSE;//负
  7. while (*str == ' ')//跳过空格
  8. {
  9. str++;
  10. }
  11. if (*str == '+' || *str == '-')
  12. {
  13. if (*str == '-')
  14. isMinus = TRUE;
  15. str++;
  16. }
  17. else if (*str < '0' || *str > '9')//如果第1位既不是符号也不是数字,直接返回异常值
  18. return 0;
  19. while (*str != '\0' && *str >= '0' && *str <= '9')
  20. {
  21. s = s * 10 + *str - '0';
  22. str++;
  23. }
  24. return s * (isMinus ? -1 : 1);
  25. }
  26. //测试
  27. int main(void)
  28. {
  29. char str[] = "86281102005404";
  30. printf("str=%lld\r\n", my_atoll(str));
  31. return 0;
  32. }

已验证:(以上代码在VS2019上运行测试,在VC++6.0上不支持%lld打印)
在这里插入图片描述
然后要把数字转成十六进制字符,依然可以使用sprintf

  1. #include <stdio.h>
  2. typedef enum { FALSE = 0, TRUE = !FALSE } BOOL;
  3. __int64 my_atoll(char* str)
  4. {
  5. __int64 s = 0;
  6. BOOL isMinus = FALSE;//负
  7. while (*str == ' ')//跳过空格
  8. {
  9. str++;
  10. }
  11. if (*str == '+' || *str == '-')
  12. {
  13. if (*str == '-')
  14. isMinus = TRUE;
  15. str++;
  16. }
  17. else if (*str < '0' || *str > '9')//如果第1位既不是符号也不是数字,直接返回异常值
  18. return 0;
  19. while (*str != '\0' && *str >= '0' && *str <= '9')
  20. {
  21. s = s * 10 + *str - '0';
  22. str++;
  23. }
  24. return s * (isMinus ? -1 : 1);
  25. }
  26. int main(void)
  27. {
  28. char str[] = "86277343909020";
  29. char strhex[100] = { 0};
  30. long long data=0;
  31. data = my_atoll(str);
  32. printf("str=%lld\r\n",data);
  33. if(data>0xFFFFFFFF)
  34. {
  35. sprintf(strhex,"%X%08X", (unsigned int)(data>>32), (unsigned int)data);
  36. }
  37. else
  38. {
  39. sprintf(strhex,"%X", (unsigned int)data);
  40. }
  41. printf("strhex=%s\r\n",strhex);
  42. return 0;
  43. }

运行结果:
在这里插入图片描述
注意,当数据大于0xFFFFFFFF,sprintf中%08X是必要的,不要写成%X,因为0需要打出来。

方案三:
接上面的方案二,my_atoll还是要用,只是将数字转成十六进制字符串可以自己实现。

  1. void ltoh(long long x, char* p)
  2. {
  3. int div;
  4. int cnt = 0;
  5. if (x == 0) {
  6. *p = '0';
  7. }
  8. if (x < 0) {
  9. x = -x;
  10. *p = '-';
  11. p++;
  12. }
  13. for (int i = 0; x != 0; i++) {
  14. div = x % 16;
  15. x = x / 16;
  16. if ((div - 10) < 0)
  17. *(p + i) = div + '0';
  18. else *(p + i) = 'A' + div - 10;
  19. div = 0;
  20. cnt++;
  21. }
  22. //reverse颠倒
  23. for (int i = 0; i < cnt / 2; i++) {
  24. char temp = p[i];
  25. p[i] = p[cnt - 1 - i];
  26. p[cnt - 1 - i] = temp;
  27. }
  28. }
  29. int main(void)
  30. {
  31. char str[] = "86281102005404";
  32. char strhex[100] = { 0 };
  33. long long data = 0;
  34. data = my_atoll(str);
  35. printf("str=%lld\r\n", data);
  36. ltoh(data,strhex);
  37. printf("strhex=%s\r\n", strhex);
  38. return 0;
  39. }

my_atoll函数见方案二。
运行结果:
在这里插入图片描述
如果比0xFFFF FFFF FFFF FFFF更大的数要转成十六进制字符串,就不在这里讨论了

PS.
vc++6.0由于版本过于陈旧,不支持long long,可以改用__int64来定义
报错:error C2632: ‘long’ followed by ‘long’ is illegal
微软最终在Visual Studio 2013上增加了对long long和unsigned long long的支持。

发表评论

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

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

相关阅读