1

我编写了一个处理 IPv6 地址的程序,我需要将四整数数组转换为表示 IPv6 地址数的十进制字符串的代码[1]。现在我遇到了需要进行反向操作的情况:将表示为单个字符串的可能较大的数字(不适合uint32_tor uint64_t,所以atol()or在这里无能为力)转换为四整数或( ) 数组(我将函数从 [ 1] 从 4-int 到变量-uint8_t)。strtol()byteuint8_t

有没有办法做到这一点?

[1]如何在 C 中将 128 位整数转换为十进制 ascii 字符串?

谢谢。这是我在这里的第一个问题,如果有的话,很抱歉英语不好。

4

1 回答 1

0

这是推荐使用库时的情况。

使用GMP,您可以进行以下转换 (gmp.c):

#include <stdio.h>
#include <stdlib.h>
#include <gmp.h>

int main(int argc, char *argv)
{
  // From string
  char *s ="199999999999999999999999999999999999999";
  mpz_t i;
  // To big number, using GMP
  // 10 here means base 10.
  mpz_init_set_str(i, s, 10);
  // Just to test if s is 128 bits long
  printf("%s can be represented with %d digits in base 2\n",s, mpz_sizeinbase(i, 2));
  // Then you can print it in hex
  gmp_printf("%#32ZX\n", i);
  // Or convert it back to int[4]
  unsigned int a[4];
  mpz_export(&a, NULL, 1, 4, 0, 0, i);
  for(int x=0;x<4;x++)
      printf("%X\n", a[x]); 
  mpz_clear(i);
  return 0;
} 

输出:

199999999999999999999999999999999999999 can be represented with 128 digits in base 2
0X96769950B50D88F41314447FFFFFFFFF
96769950
B50D88F4
1314447F
FFFFFFFF

我在 32 位 Linux 系统上测试了这段代码。请注意不同平台上不同的 int 大小和字节序。您可能想要大端的结果,如果是这样,只需将 mz_export 更改为:

mpz_export(&a, NULL, 1, 4, 1, 0, i);

要编译示例,不要忘记安装 gmp 并在 gcc 命令行参数上添加 -lgmp -std=c99。

在 Ubuntu 上,您可以使用以下命令安装 gmp:

sudo apt-get install libgmp-dev

并编译示例 gmp.c:

gcc gmp.c -o gmp -lgmp -std=c99

此代码可以成为您转换的良好开端。您还可以将 i 初始化为固定的 128 位数字(使用 GMP 的 init 函数),以防万一您的大数字以零开头。

于 2012-10-09T10:04:16.273 回答