将十六进制字符串转换为数据

3
我发现了许多不同的解决方案,但并非所有解决方案都有效,而且许多解决方案似乎有些繁琐和低效。基本上,我有一串十六进制数据(即“55 AA 41 2A 00 94 55 AA BB BB 00 FF”),希望将其转换为原始数据。最好的方法是什么? 更新:Vicky的解决方案对我很有效,但我对其进行了修改以处理没有空格的十六进制字符串,并稍微改变了风格。
int i = 0;
char *hexString = "55AA412A009455AABBBB00FF"
char *hexPtr = hexString;
unsigned int *result = calloc(strlen(hexString)/2 + 1, sizeof *result);

while (sscanf(hexPtr, "%02x", &result[i++])) {
    hexPtr += 2;
    if (hexPtr >= hexString + strlen(hexString)) break;
}

return result;

你能简要澄清一下,你尝试过的几种方法以及这段代码适用于什么类型的上下文吗? - nielsj
5
可能重复:[代码高尔夫-十六进制到(原始)二进制转换](https://dev59.com/AnRA5IYBdhLWcg3w4SDo) - zaf
zaf,这正是我想要的,很抱歉我之前没看到。我猜这在某种程度上算是那个的重复。 - AriX
不需要将&result[i++]强制转换为(unsigned int *) - 它已经具有该类型。此外,您未为result分配足够的空间 - 您需要在calloc调用中将最后一个, 1)替换为sizeof *result - caf
5个回答

2

这个字符串的长度始终相同吗?

如果是这样:

char *buf = "55 AA 41 2A 00 94 55 AA BB BB 00 FF";
sscanf(buf, "%x %x %x [repeat as many times as necessary]", &a, &b, &c [etc]);

如果不是:
#include <stdio.h>
#include <string.h>
#include <stdlib.h>

int main (int argc, char ** argv)
{
    char *buf = "55 AA 41 2A 00 94 55 AA BB BB 00 FF";
    char *p = buf;
    int i = 0, j;
    unsigned int *result = calloc(strlen(buf)/3 + 1 * sizeof(int), 1);

    if (result)
    {
        while (sscanf(p, "%02x", &result[i++]))
        {
             p += 3;
            if (p > buf + strlen(buf))
            {
             break;
            }
        }

        printf("%s\n", buf);

        for (j = 0; j < i; j++)
        {
            printf("%02X ", result[j]);
        }

        printf("\n");
    }
}

%02x 转换说明符需要一个指向 unsigned int 的指针,但您正在传递一个指向 unsigned char 的指针。这会导致未定义的行为(实际上,在不是小端机器上,您的答案将产生不正确的结果,并且在具有严格对齐要求的机器上崩溃)。 - caf

1
你尝试过 sscanf 或者 scanf 吗?这个函数可以处理十六进制值并返回“原始数据”。

0
 /*
  * copyLeft (ButterC) All Right Changed :)
 */
 void _StackOverFlow(void *data) {

    unsigned char *ButterC = (unsigned char *) data;
    size_t ssize ,size = ssize = 0; 

    int i;

    for (i = 0; ButterC[i] != 0; i++)
    {
            if (ButterC[i] != 32)
                    size++;
    }

    unsigned int *resu = (unsigned int *) malloc((size / 2) + 1);


    while (*ButterC != 0 && resu != NULL && ssize < size) 
    {
            if (*ButterC == 32) 
                    ButterC++;
            else {
                    sscanf(ButterC,"%02x",&resu[ssize++]); ButterC += 2;
            }
    }
    // TEST RESULTE TO => (resu) 
    for (i = 0; i < ssize; i++)
    {
            printf("%02X ",resu[i] & 0xff);
    }

    puts(""); // NEWLINE  
}

int _start() {

    char data[] = "FF 00FF           01";
    _StackOverFlow(data);

    
    _exit(0);
}

0
#define GETBITS(a) (a < 'A' ? a - '0' : toupper(a) - 'A' + 10)

char *cur;
char data;
for (cur = buffer; *cur; cur++) {
    data = GETBITS(cur[0]) << 4 + GETBITS(cur[1]);
    cur += 2;
    printf("%c", data);
}
printf("\n");

0
while(input[i]!='\0'){        
    *(temp++) = ((input[i]>='0' && input[i]<='9')?(input[i]-'0'):(input[i]-'a'+10))*16 +( (input[i+1]>='0' && input[i]<='9')?(input[i+1]-'0'):(input[i+1]-'a'+10));
         i+=2;
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接