Преобразование строки чисел в массив числовых байтов в C - PullRequest
0 голосов
/ 29 января 2019

Я хотел бы преобразовать строку, содержащую только целые числа, в массив байтов, но для эффективного хранения (поэтому нет «digits [digitIndex] = string [digitIndex - '0';»).Я хотел бы, чтобы они были сохранены так же, как и любой другой тип: они имеют 256 различных возможностей на байт, а не только 10, как в предыдущем, ошибочном примере.Он также должен содержать много цифр (в качестве размера я использую 8-битный параметр, поэтому я считаю, что он должен содержать не менее 100 цифр).Изменить: я также не хочу использовать какие-либо библиотеки по личным причинам.

Вот пример того, как это будет выглядеть в функции:

int8_t *stringToBigInt(char *input) {
    uint8_t digitsBase10 = strlen(input);
    uint8_t bytes = ???; //However many bytes to store the result (max 255 bytes in this case)
    int8_t *result = malloc(sizeof(void *) + bytes);
    ... //Code for setting result to input
    return result;
}

А вот примервозможный ввод и вывод:

Редактировать: это короткий пример, который вписывается в 32-битные только для простоты;вход может быть намного больше, чем 32-битное (и, возможно, 64-битное) целое число

Вход: «1234567890»

Выход: {01001001, 10010110, 00000010, 11010010}

1 Ответ

0 голосов
/ 29 января 2019

Это базовое преобразование из base-10 в base-256, так что это то, что вы должны искать, насколько далеко идут алгоритмы.Для упрощенной реализации сначала реализуем длинное деление на степени 2, работающее со строками.Затем преобразуйте каждый из оставшихся в байт: эти байты формируют ваш вывод.Вы захотите многократно разделить входные данные, и каждая строка из 8 оставшихся битовых остатков образует 256-байтовые байты, начиная с наименее значащей цифры (один байт - одна 256-битная цифра).Повторное деление означает, что вы передаете частное от предыдущего деления последующему, как дивиденд.

Существуют некоторые классные алгоритмы, которые могут делить числа base-10 на степени два, которые работают намного быстрее ипроще, чем обобщенное длинное деление.В качестве подсказки, давайте возьмем пример: 510. Мы делим каждую цифру на две и передаем остаток * 5 следующей цифре.Давайте отбросим дробную часть меньше 0,5: 510 станет 2 * 100 + 5 * 10 + 5. Затем 1 * 100 + 2 * 10 + 2, точка 5. Затем 6 * 10 + 1. Затем 3 * 10, точка 5, 2 *.10 + 5, затем 1 * 10 + 2 точка 5, затем 6, затем 3, затем 2 точка 5, затем 1, затем 0 точка 5.

Для 255 мы получили бы 127,5, 63,5, 15,5,7,5, 3,5, 1,5, 0,5.

Деление на более высокие коэффициенты двух возможно, но требует многократных длинных дополнений.Например, 33 div 4 = 0 * 10 + 7rem1 + 0 rem 0,75 (га!).Деление на два работает лучше, так как мы используем тот факт, что 10 = 2 * 5, и обозначение base-n можно легко разделить на коэффициенты основания, не выполняя длинных сложений: все операции ограничены двумя соседними цифрами, поэтому это линейноевремя процесса со стоимостью N в количестве цифр.Но для преобразования базы в base-256 вы делаете повторное деление, поэтому стоимость составляет ~ 0.5N ^ 2.Легко реализовать, но дорого в вычислениях.

Конечно, есть и лучшие алгоритмы.Но все вышесказанное можно реализовать кратко - даже в форме библиотечных функций довольно хорошего качества:

Сначала давайте определим тип массива байтов и способ передачи его в читаемый человеком шестнадцатеричный вывод.Для удобства объект указывается через указатель на его данные, а детали реализации вообще не фигурируют в интерфейсе.Конструктор new_Bytes инициализирует массив нулем.Существует также метод, который обрабатывает массив так, как если бы он был массивом битов, упорядоченным по порядку номерам (сначала LSB) и устанавливает (включает) данный бит.

// https://github.com/KubaO/stackoverflown/tree/master/questions/decimal-to-binary-54422895
#include <assert.h>
#include <inttypes.h>
#include <stdbool.h>
#include <stddef.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

// Bytes Class Interface

typedef uint8_t *Bytes;
typedef const uint8_t *cBytes;

Bytes new_Bytes(size_t size);
size_t Bytes_size(cBytes bytes);
void Bytes_truncate(Bytes bytes, size_t new_size);
void free_Bytes(cBytes bytes);
char *Bytes_to_hex(cBytes bytes);

static inline void Bytes_set_bit(Bytes const bytes, size_t const bit_num) {
   bytes[bit_num / 8] |= 1 << (bit_num % 8);
}

Затем деление-by-2 выполняется на месте, а флаги предоставляют дополнительную информацию, необходимую для преобразования базы, особенно остаток.Преобразование из базы 10 в базу 256 использует деление и возвращает новый массив Bytes.

// Division and Base Conversion Interface

typedef enum {
   REMAINDER = 1,           /* there is a non-zero remainder */
   ZERO = 2,                /* the quotient is zero or null */
   NULL_DECIMAL = 4,        /* the dividend is null or empty */
   NON_DECIMALS = 8,        /* division was terminated on non-decimal characters */
   LEADING_ZERO_COUNT = 16, /* count of leading zeroes in the quotient */
   LEADING_ZERO_COUNT_MASK = ~(LEADING_ZERO_COUNT - 1),
   CLR_CARRY_MASK = ~REMAINDER,
   CLR_ZERO_MASK = ~ZERO,
} DivFlags;

DivFlags divide_by_2(char *decimal);
Bytes base_10_to_256(const char *decimal);

Деление работает с десятичным представлением в порядке от старшей к младшей значащей цифры.Каждая цифра объединяется с остатком от деления предыдущей цифры, а затем делится на 2. Остаток переносится между делениями цифр.После деления наименьшей значащей цифры остаток выводится во флаги.

Флаги в основном не требуют пояснений, но LEADING_ZERO_COUNT не совсем - и, следовательно, доступ к ним осуществляется через функции доступа,LEADING_ZERO_COUNT является единицей подсчета ведущих нулей.По мере того, как деление проходит через десятичное представление, оно будет считать начальные нули, умножить их на эту единицу и объединить с флагами.Чтобы извлечь количество, флаги делятся на единицы.

// Division and Base Conversion Implementation

static inline int leading_zero_count(DivFlags const flags) {
   return (flags & LEADING_ZERO_COUNT_MASK) / LEADING_ZERO_COUNT;
}

static inline void saturated_inc_leading_zero_count(DivFlags *flags) {
   if ((*flags & LEADING_ZERO_COUNT_MASK) != LEADING_ZERO_COUNT_MASK)
      *flags += LEADING_ZERO_COUNT;
}

DivFlags divide_by_2(char *decimal) {
   DivFlags flags = ZERO;
   if (!decimal) return flags | NULL_DECIMAL;
   char c;
   while ((c = *decimal)) {
      if (c < '0' || c > '9') return flags | NON_DECIMALS;
      c = c - '0' + ((flags & REMAINDER) ? 10 : 0);
      if (c & 1)
         flags |= REMAINDER;
      else
         flags &= CLR_CARRY_MASK;
      c >>= 1;
      assert(c >= 0 && c <= 9);
      if (c)
         flags &= CLR_ZERO_MASK;
      else if (flags & ZERO)
         saturated_inc_leading_zero_count(&flags);
      *decimal++ = c + '0';
   }
   return flags;
}

Затем базовое преобразование выполняет повторное деление на 2 и сдвигает оставшиеся биты в байтовый массив следующим образом:

Сначала базовое преобразование берет копию десятичного представления и выделяет выходной байтовый массив соответствующего размера.

static void base_10_to_256_impl(Bytes const bytes, char *decimal);

Bytes base_10_to_256(const char *const decimal) {
   assert(decimal);
   size_t const dec_len = strlen(decimal);
   char *const dec_buf = malloc(dec_len + 1);
   if (!dec_buf) return NULL;
   memcpy(dec_buf, decimal, dec_len + 1);

   size_t const BASE_RATIO_NUM = 416, /* ceil(log(10)/log(256)*1000) */
       BASE_RATIO_DENOM = 1000;
   assert(dec_len <= (SIZE_MAX / BASE_RATIO_NUM));
   size_t const len = (size_t)(dec_len * BASE_RATIO_NUM / BASE_RATIO_DENOM) + 1;
   Bytes const bytes = new_Bytes(len);  // little-endian
   if (bytes) base_10_to_256_impl(bytes, dec_buf);
   free(dec_buf);
   return bytes;
}

Затем, в «мясе» реализации, функция выполняет итерациювыходные биты, многократно разделяющие десятичное представление на 2, и устанавливающие каждый бит со значением оставшегося бита.

static void base_10_to_256_impl(Bytes const bytes, char *decimal) {
   size_t const len = Bytes_size(bytes);
   for (size_t bit_num = 0;; bit_num++) {
      DivFlags const flags = divide_by_2(decimal);
      assert(!(flags & NULL_DECIMAL));
      decimal += leading_zero_count(flags);
      if (flags & ZERO && !(flags & REMAINDER)) {
         size_t const new_len = ((bit_num + 7) / 8);
         Bytes_truncate(bytes, new_len);
         break;
      }
      // here, there are still non-zero bits - in the dec[imal] and/or in the carry
      assert((bit_num / 8) < len);
      if (flags & REMAINDER) Bytes_set_bit(bytes, bit_num);
   }
}

Теперь мы можем добавить несколько тестов:

// Tests

void check_bytes(const char *const decimal, const char *const bytes_expected,
                 size_t const bytes_len, const char *const hex_expected) {
   cBytes const bytes = base_10_to_256(decimal);
   assert(bytes && Bytes_size(bytes) == bytes_len);
   assert(memcmp(bytes, bytes_expected, bytes_len) == 0);
   char *const hex = Bytes_to_hex(bytes);
   assert(hex && strcmp(hex, hex_expected) == 0);
   printf("%s\n", hex);
   free(hex);
   free_Bytes(bytes);
}

int main() {
   check_bytes("4294967297" /* 2^32+1 */, "\1\0\0\0\1", 5, "01 00000001");
   check_bytes("4294967296" /* 2^32   */, "\0\0\0\0\1", 5, "01 00000000");
   check_bytes("4294967295" /* 2^32-1 */, "\xFF\xFF\xFF\xFF", 4, "FFFFFFFF");
   check_bytes("16777217" /* 2^24+1 */, "\1\0\0\1", 4, "01000001");
   check_bytes("16777216" /* 2^24   */, "\0\0\0\1", 4, "01000000");
   check_bytes("16777215" /* 2^24-1 */, "\xFF\xFF\xFF", 3, "FFFFFF");
   check_bytes("256", "\0\1", 2, "0100");
   check_bytes("255", "\xFF", 1, "FF");
   check_bytes("254", "\xFE", 1, "FE");
   check_bytes("253", "\xFD", 1, "FD");
   check_bytes("3", "\3", 1, "03");
   check_bytes("2", "\2", 1, "02");
   check_bytes("1", "\1", 1, "01");
   check_bytes("0", "\0", 1, "00");
}

ImПлементация класса Bytes завершает пример:

// Bytes Implementation

struct BytesImpl {
   size_t size;
   uint8_t data[1];
};
static const size_t Bytes_header_size = offsetof(struct BytesImpl, data);
_Static_assert(offsetof(struct BytesImpl, data) == sizeof(size_t),
               "unexpected layout of struct BytesImpl");

Bytes new_Bytes(size_t size) {
   assert(size <= SIZE_MAX - Bytes_header_size);
   if (!size) size++;
   struct BytesImpl *const impl = calloc(Bytes_header_size + size, 1);
   if (!impl) return NULL;
   impl->size = size;
   return &impl->data[0];
}

static const struct BytesImpl *Bytes_get_const_impl_(cBytes const bytes) {
   return (const struct BytesImpl *)(const void *)((const char *)bytes -
                                                   Bytes_header_size);
}

static struct BytesImpl *Bytes_get_impl_(Bytes const bytes) {
   return (struct BytesImpl *)(void *)((char *)bytes - Bytes_header_size);
}

size_t Bytes_size(cBytes const bytes) { return Bytes_get_const_impl_(bytes)->size; }

void Bytes_truncate(Bytes const bytes, size_t new_size) {
   size_t *const size = &Bytes_get_impl_(bytes)->size;
   if (!new_size) {
     new_size++;  // we always leave one byte in the array
     bytes[0] = 0;
   }
   assert(*size);
   if (*size <= new_size) return;
   *size = new_size;
}

void free_Bytes(cBytes const bytes) {
   if (bytes) free((void *)(intptr_t)(const void *)Bytes_get_const_impl_(bytes));
}

char *Bytes_to_hex(cBytes const bytes) {
   size_t n = Bytes_size(bytes);
   size_t spaces = (n - 1) / 4;
   char *const out = malloc(n * 2 + spaces + 1);
   if (out)
      for (char *o = out; n;) {
         uint8_t const c = bytes[n - 1];
         snprintf(o, 3, "%02" PRIX8, c);
         o += 2;
         n--;
         if (n && n % 4 == 0) {
            assert(spaces);
            *o++ = ' ';
            spaces--;
         }
      }
   return out;
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...