十六进制字符数组用C(Hex to char array in C)

2019-07-22 17:03发布

鉴于十六进制值如“0011223344”的字符串,所以这是为0x00,为0x11等。

如何将这些值添加到一个字符数组?

相当于说:

char array[4] = { 0x00, 0x11 ... };

Answer 1:

不能容纳5个字节的数据值得到一个4字节数组; 导致缓冲区溢出。

如果你有十六进制数字的字符串,你可以使用sscanf()和一个循环:

#include <stdio.h>
#include <ctype.h>

int main()
{
    const char *src = "0011223344";
    char buffer[5];
    char *dst = buffer;
    char *end = buffer + sizeof(buffer);
    unsigned int u;

    while (dst < end && sscanf(src, "%2x", &u) == 1)
    {
        *dst++ = u;
        src += 2;
    }

    for (dst = buffer; dst < end; dst++)
        printf("%d: %c (%d, 0x%02x)\n", dst - buffer,
               (isprint(*dst) ? *dst : '.'), *dst, *dst);

    return(0);
}

需要注意的是印刷开始一个零字节字符串需要照顾; 大多数操作上的第一个空字节终止。 请注意,此代码没有空终止缓冲区; 目前尚不清楚空终止是否是期望的,并没有在我声明添加一个终端空缓冲足够的空间(但是易于固定的)。 有一个像样的机会,如果代码被封装成一个子程序,那就需要返回转换后的字符串的长度(虽然你也可以认为这是源字符串除以二的长度)。



Answer 2:

我会做这样的事情;

// Convert from ascii hex representation to binary
// Examples;
//   "00" -> 0
//   "2a" -> 42
//   "ff" -> 255
// Case insensitive, 2 characters of input required, no error checking
int hex2bin( const char *s )
{
    int ret=0;
    int i;
    for( i=0; i<2; i++ )
    {
        char c = *s++;
        int n=0;
        if( '0'<=c && c<='9' )
            n = c-'0';
        else if( 'a'<=c && c<='f' )
            n = 10 + c-'a';
        else if( 'A'<=c && c<='F' )
            n = 10 + c-'A';
        ret = n + ret*16;
    }
    return ret;
}

int main()
{
    const char *in = "0011223344";
    char out[5];
    int i;

    // Hex to binary conversion loop. For example;
    // If in="0011223344" set out[] to {0x00,0x11,0x22,0x33,0x44}
    for( i=0; i<5; i++ )
    {
        out[i] = hex2bin( in );
        in += 2;
    }
    return 0;
}


Answer 3:

如果字符串是正确的,并没有必要保留它的内容,那么我会做这种方式:

#define hex(c) ((*(c)>='a')?*(c)-'a'+10:(*(c)>='A')?*(c)-'A'+10:*(c)-'0') 

void hex2char( char *to ){
  for(char *from=to; *from; from+=2) *to++=hex(from)*16+hex(from+1);
  *to=0;
}

编辑1:对不起,我忘记了用字母AF计算(AF)

编辑2:我试着写一个更迂腐的代码:

#include <string.h> 

int xdigit( char digit ){
  int val;
       if( '0' <= digit && digit <= '9' ) val = digit -'0';
  else if( 'a' <= digit && digit <= 'f' ) val = digit -'a'+10;
  else if( 'A' <= digit && digit <= 'F' ) val = digit -'A'+10;
  else                                    val = -1;
  return val;
}

int xstr2str( char *buf, unsigned bufsize, const char *in ){
  if( !in ) return -1; // missing input string

  unsigned inlen=strlen(in);
  if( inlen%2 != 0 ) return -2; // hex string must even sized

  for( unsigned i=0; i<inlen; i++ )
    if( xdigit(in[i])<0 ) return -3; // bad character in hex string

  if( !buf || bufsize<inlen/2+1 ) return -4; // no buffer or too small

  for( unsigned i=0,j=0; i<inlen; i+=2,j++ )
    buf[j] = xdigit(in[i])*16 + xdigit(in[i+1]);

  buf[inlen/2] = '\0';
  return inlen/2+1;
}

测试:

#include <stdio.h> 

char buf[100] = "test";

void test( char *buf, const char *s ){
   printf("%3i=xstr2str( \"%s\", 100, \"%s\" )\n", xstr2str( buf, 100, s ), buf, s );
}

int main(){
  test( buf,      (char*)0   );
  test( buf,      "123"      );
  test( buf,      "3x"       );
  test( (char*)0, ""         );
  test( buf,      ""         );
  test( buf,      "3C3e"     );
  test( buf,      "3c31323e" );

  strcpy( buf,    "616263"   ); test( buf, buf );
}

结果:

 -1=xstr2str( "test", 100, "(null)" )
 -2=xstr2str( "test", 100, "123" )
 -3=xstr2str( "test", 100, "3x" )
 -4=xstr2str( "(null)", 100, "" )
  1=xstr2str( "", 100, "" )
  3=xstr2str( "", 100, "3C3e" )
  5=xstr2str( "", 100, "3c31323e" )
  4=xstr2str( "abc", 100, "abc" )


Answer 4:

比方说,这是一个little-endian的ASCII平台。 也许OP的意思是“字符数组”,而不是“字符串”。我们有对焦炭的工作和位屏蔽。注X16的shiftyness ..

/* not my original work, on stacko somewhere ? */

for (i=0;i < 4;i++) {

    char a = string[2 * i];
    char b = string[2 * i + 1];

    array[i] = (((encode(a) * 16) & 0xF0) + (encode(b) & 0x0F));
 }

和函数编码()被定义...

unsigned char encode(char x) {     /* Function to encode a hex character */
/****************************************************************************
 * these offsets should all be decimal ..x validated for hex..              *
 ****************************************************************************/
    if (x >= '0' && x <= '9')         /* 0-9 is offset by hex 30 */
        return (x - 0x30);
    else if (x >= 'a' && x <= 'f')    /* a-f offset by hex 57 */
        return(x - 0x57);
    else if (x >= 'A' && x <= 'F')    /* A-F offset by hex 37 */
        return(x - 0x37);
}

这种方法彩车周围其他地方,这不是我的原创作品,但它是旧的。 由纯粹主义者不喜欢,因为它是不可移植的,但扩展将是微不足道的。



Answer 5:

Fatalfloor ...

有一对夫妇的方式做到这一点...首先,你可以使用memcpy()来精确表示复制到字符数组。

你可以用位移位和位屏蔽技术,以及。 我猜这就是你需要做的,因为它听起来像一个家庭作业的问题是什么。

最后,你可以使用一些花哨的指针间接复制你需要的内存位置。

所有这些方法都在这里详细介绍:

存放在字符数组一个int?



Answer 6:

举一个最好的方法:

十六进制字符串到数字值,即STR [] =“0011223344”珍惜0x0011223344,使用

value = strtoul(string, NULL, 16); // or strtoull()

完成。 如果需要删除开始为0x00,见下文。

尽管对于LITTLE_ENDIAN平台,以及:十六进制数值为char数组,值0x11223344到char ARR [N] = {0×00,0×11,...}

unsigned long *hex = (unsigned long*)arr;
*hex = htonl(value);
// you'd like to remove any beginning 0x00
char *zero = arr;
while (0x00 == *zero) { zero++; }
if (zero > arr) memmove(zero, arr, sizeof(arr) - (zero - arr));

完成。

注:对于长字符串转换为64位十六进制数字的常用3在32位系统上,你应该使用无符号长长的,而不是无符号长,htonl是不够的,所以下面自己做,因为可能没有htonll,htonq或hton64等:

#if __KERNEL__
    /* Linux Kernel space */
    #if defined(__LITTLE_ENDIAN_BITFIELD)
        #define hton64(x)   __swab64(x)
    #else
        #define hton64(x)   (x)
    #endif
#elif defined(__GNUC__)
    /* GNU, user space */
    #if __BYTE_ORDER == __LITTLE_ENDIAN 
        #define hton64(x)   __bswap_64(x)
    #else
        #define hton64(x)   (x)
    #endif
#elif 
         ...
#endif

#define ntoh64(x)   hton64(x)

看到http://effocore.googlecode.com/svn/trunk/devel/effo/codebase/builtin/include/impl/sys/bswap.h



Answer 7:

{
    char szVal[] = "268484927472";
    char szOutput[30];

    size_t nLen = strlen(szVal);
    // Make sure it is even.
    if ((nLen % 2) == 1)
    {
        printf("Error string must be even number of digits %s", szVal);
    }

    // Process each set of characters as a single character.
    nLen >>= 1;
    for (size_t idx = 0; idx < nLen; idx++)
    {
        char acTmp[3];
        sscanf(szVal + (idx << 1), "%2s", acTmp);
        szOutput[idx] = (char)strtol(acTmp, NULL, 16);
    }
}


Answer 8:

我在寻找同样的事情,读了很多,终于创造了这个功能。 认为这可能帮助别人

// in = "63 09  58  81" 
void hexatoascii(char *in, char* out, int len){
    char buf[5000];
    int i,j=0;
    char * data[5000];
    printf("\n size %d", strlen(in));
    for (i = 0; i < strlen(in); i+=2)
    {
        data[j] = (char*)malloc(8);
        if (in[i] == ' '){
            i++;
        }
        else if(in[i + 1] == ' '){
            i++;
        }
        printf("\n %c%c", in[i],in[i+1]);
        sprintf(data[j], "%c%c", in[i], in[i+1]);
        j++;
    }

    for (i = 0; i < j-1; i++){
        int tmp;
        printf("\n data %s", data[i] );
        sscanf(data[i], "%2x", &tmp);
        out[i] = tmp;
    }
    //printf("\n ascii value of hexa %s", out);
}


Answer 9:

首先,你的问题是不是很精确。 是字符串std::stringchar缓冲区? 设置在编译时?

动态内存是几乎可以肯定你的答案。

char* arr = (char*)malloc(numberOfValues);

然后,您可以通过输入走,并将其分配给数组。



文章来源: Hex to char array in C
标签: c arrays char hex