C - 无法调用函数

时间:2018-04-09 11:54:29

标签: c base64

使用以下代码,我尝试将字符串编码为基数64.我无法弄清楚如何调用该方法。

我尝试将输入字符串"test"转换为base64中的"dGVzdA=="

我已尝试使用strcpy(data,txt),但无法弄清楚如何完成方法调用。

#include <stdio.h>
#include <string.h>
#include <ctype.h>
#include <stdint.h>
#include <stdlib.h>

static char encoding_table[] = {'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H',
                                'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P',
                                'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X',
                                'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f',
                                'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n',
                                'o', 'p', 'q', 'r', 's', 't', 'u', 'v',
                                'w', 'x', 'y', 'z', '0', '1', '2', '3',
                                '4', '5', '6', '7', '8', '9', '+', '/'};

static int mod_table[] = {0, 2, 1};

char *base64_encode(char *data,
                    size_t input_length,
                    size_t *output_length) {

    *output_length = 4 * ((input_length + 2) / 3);

    char *encoded_data = malloc(*output_length);
    if (encoded_data == NULL) return NULL;

    for (int i = 0, j = 0; i < input_length;) {

        uint32_t octet_a = i < input_length ? (unsigned char)data[i++] : 0;
        uint32_t octet_b = i < input_length ? (unsigned char)data[i++] : 0;
        uint32_t octet_c = i < input_length ? (unsigned char)data[i++] : 0;

        uint32_t triple = (octet_a << 0x10) + (octet_b << 0x08) + octet_c;

        encoded_data[j++] = encoding_table[(triple >> 3 * 6) & 0x3F];
        encoded_data[j++] = encoding_table[(triple >> 2 * 6) & 0x3F];
        encoded_data[j++] = encoding_table[(triple >> 1 * 6) & 0x3F];
        encoded_data[j++] = encoding_table[(triple >> 0 * 6) & 0x3F];
    }

    for (int i = 0; i < mod_table[input_length % 3]; i++)
        encoded_data[*output_length - 1 - i] = '=';

    return encoded_data;
}

int main() {
  char txt[] = "test";
  char *data;
  //... Please help 
}

如何将txt"test")转换为"dGVzdA=="并打印出来?

2 个答案:

答案 0 :(得分:0)

你应该阅读一本关于C编程的书或教程,因为调用函数是一个非常基本的概念。

要回答你的问题,请通过明文,它的长度和变量的地址来接收结果输出的长度:

int main() {
  char txt[] = "test";
  char *data;
  size_t output_length;

  data = base64_encode(txt, strlen(txt), &output_length);
  printf("%*s\n", output_length, data);
}

该函数返回指向数据的指针。在上面的代码示例中,将分配给data。返回字符串的长度将由output_length变量中的函数设置。

重要的是要注意返回的字符串 not null在base64_encode()中终止。如果您假设它是一个以空字符结尾的字符串并尝试使用它,那么这可能会导致代码中的问题。例如,天真地打印它或试图用strlen()找到它的长度可能会导致缓冲区溢出并可能导致程序崩溃。

您可以将空字符('\0')附加到data,但是,base64_encode()仅为编码字符串分配足够的空间,因此严格来说附加任何内容都可以访问无效内存导致崩溃。解决方案包括:

  • 小心处理字符串。用这样的东西打印出来:

       printf("%*s", output_length, data);
    
  • 分配您自己的长度为output_length + 1的字符串,复制 将字符串编码为新字符串,然后null终止新字符串:

       data = base64_encode(txt, strlen(txt), &output_length);
       char *encoded = malloc(output_length + 1);
       strncpy(encoded, data, output_length);
       encoded[output_length] = '\0';
    

最后,确保在完成base64编码数据后调用free(data),以确保释放函数base64_encode()中分配的内存。

答案 1 :(得分:0)

以下是main()调用base64_encode()

int main(void) {
  char data[] = "test";
  char *encoded_data;
  size_t len;

  encoded_data = base64_encode(data, strlen(data), &len);
  printf("%s (%zu) => %s (%zu)", data, strlen(data), encoded_data, len);
}

解释

  • char data[] 包含要编码的字符串。您需要将其作为参数传递给base64_encode()
  • char *encoded_data 是一个指针,您将用它来存储base64_encode()的返回值。您将使用此变量来访问已解码的字符串,因为base64_encode()返回指向已存储已解码字符串的已分配空间的指针。
  • size_t len 是一个变量,用于存储已解码字符串的大小。您将指向此变量的指针作为base64_encode()的参数。
  • strlen(data) base64_encode()的第二个参数,是char data[]的长度。