Base64 编码与解码

#include <string.h>
#include "mimeb64.h"

//---------------------------------------------------------------------------
//  4bit binary to char 0-F
char Hex2Chr( unsigned char n )
{
	n &= 0xF;
    if ( n < 10 )
    	return ( char )( n + '0' );
    else
    	return ( char )( n - 10 + 'A' );
}
//---------------------------------------------------------------------------
//  char 0-F to 4bit binary
unsigned char Chr2Hex( char c )
{
	if ( c >= 'a' && c <= 'z' )  //  it's toupper
    	c = c - 'a' + 'A';
    if ( c >= '0' && c <= '9' )
    	return ( int )( c - '0' );
    else if ( c >= 'A' && c <= 'F' )
    	return ( int )( c - 'A' + 10 );
    else
    	return -1;
}
//---------------------------------------------------------------------------
//  Base64 code table
//  0-63 : A-Z(25) a-z(51), 0-9(61), +(62), /(63)
char  Base2Chr( unsigned char n )
{
	n &= 0x3F;
	if ( n < 26 )
    	return ( char )( n + 'A' );
    else if ( n < 52 )
    	return ( char )( n - 26 + 'a' );
    else if ( n < 62 )
    	return ( char )( n - 52 + '0' );
    else if ( n == 62 )
    	return '+';
    else
    	return '/';
}
//---------------------------------------------------------------------------
unsigned char Chr2Base( char c )
{
	if ( c >= 'A' && c <= 'Z' )
    	return ( unsigned char )( c - 'A' );
    else if ( c >= 'a' && c <= 'z' )
    	return ( unsigned char )( c - 'a' + 26 );
    else if ( c >= '0' && c <= '9' )
    	return ( unsigned char )( c - '0' + 52 );
    else if ( c == '+' )
    	return 62;
    else if ( c == '/' )
    	return 63;
    else
        return 64;  //  无效字符
}
//---------------------------------------------------------------------------
//  aLen 为 aSrc 的大小, aDest 所指的缓冲区必须至少为 aLen 的 3 倍!!!
//  返回 aDest 的长度
int QPEncode( char * const aDest, const unsigned char * aSrc, int aLen )
{
	char * p = aDest;
    int    i = 0;

    while ( i++ < aLen )
    {
    	*p++ = '=';
        *p++ = Hex2Chr( *aSrc >> 4 );
        *p++ = Hex2Chr( *aSrc++ );
    }
    *p = 0;  //  aDest is an ASCIIZ string
	return ( p - aDest );  //  exclude the end of zero
}
//---------------------------------------------------------------------------
//  aDest 所指的缓冲区必须至少为 aSrc 长度的 1/3 !!!
//  返回 aDest 的长度
int QPDecode( unsigned char * const aDest, const char * aSrc )
{
	unsigned char * p = aDest;
    int             n = strlen( aSrc );
    unsigned char   ch, cl;

    while ( *aSrc )  //  aSrc is an ASCIIZ string
    {
    	if ( ( *aSrc == '=' ) && ( n - 2 > 0 ) )
        {
        	ch = Chr2Hex( aSrc[1] );
            cl = Chr2Hex( aSrc[2] );
            if ( ( ch == ( unsigned char )-1 ) || ( cl == ( unsigned char )-1 ) )
            	*p++ = *aSrc++;
            else
            {
				*p++ = ( ch << 4 ) | cl;
    	        aSrc += 3;
            }
        }
        else
        	*p++ = *aSrc++;
    }
	return ( p - aDest );
}
//---------------------------------------------------------------------------
//  aLen 为 aSrc 的长度, aDest 所指的缓冲区必须至少为 aLen 的 4/3 倍!!!
//  返回 aDest 的长度
int Base64Encode( char * const aDest, const unsigned char * aSrc, int aLen )
{
	char        * p = aDest;
    int           i;
    unsigned char t;

    for ( i = 0; i < aLen; i++ )
    {
    	switch ( i % 3 )
        {
        case 0 :
        	*p++ = Base2Chr( *aSrc >> 2 );
            t = ( *aSrc++ << 4 ) & 0x3F;
            break;
        case 1 :
        	*p++ = Base2Chr( t | ( *aSrc >> 4 ) );
            t = ( *aSrc++ << 2 ) & 0x3F;
            break;
        case 2 :
        	*p++ = Base2Chr( t | ( *aSrc >> 6 ) );
            *p++ = Base2Chr( *aSrc++ );
            break;
        }
    }
    if ( aLen % 3 != 0 )
    {
    	*p++ = Base2Chr( t );
        if ( aLen % 3 == 1 )
        	*p++ = '=';
        *p++ = '=';
    }
    *p = 0;  //  aDest is an ASCIIZ string
	return ( p - aDest );  //  exclude the end of zero
}
//---------------------------------------------------------------------------
//  aDest 所指的缓冲区必须至少为 aSrc 长度的 0.75 倍!!!
//  返回 aDest 的长度
int Base64Decode( unsigned char * const aDest, const char * aSrc )
{
	unsigned char * p = aDest;
    int             i, n = strlen( aSrc );
    unsigned char   c, t;

    for ( i = 0; i < n; i++ )
    {
    	if ( *aSrc == '=' )
        	break;
        do {
        	if ( *aSrc )
		       	c = Chr2Base( *aSrc++ );
            else
            	c = 65;  //  字符串结束
        } while ( c == 64 );  //  跳过无效字符,如回车等
        if ( c == 65 )
        	break;
        switch ( i % 4 )
        {
        case 0 :
            t = c << 2;
            break;
        case 1 :
            *p++ = ( unsigned char )( t | ( c >> 4 ) );
            t = ( unsigned char )( c << 4 );
            break;
        case 2 :
            *p++ = ( unsigned char )( t | ( c >> 2 ) );
            t = ( unsigned char )( c << 6 );
            break;
        case 3 :
            *p++ = ( unsigned char )( t | c );
            break;
        }
    }
	return ( p - aDest );
}
//---------------------------------------------------------------------------

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1. 编码 Base64是一种用64个字符来表示任意二进制数据的编码方式。编码过程就是将二进制数据转换为可打印的ASCII字符集中的字符。 编码的过程如下: - 将传入的数据按照每3个字节一组进行划分。 - 对于每一组,将其转换为4个6位的数字,即24位。 - 将这4个数字转换为4个ASCII字符,即每个字符6位,不足6位的在后面补0。 - 如果原数据的长度不是3的倍数,则在末尾补0,然后用“=”字符表示补了几个0。 C语言中可以使用标准库中的函数实现Base64编码,具体实现如下: #include <stdio.h> #include <stdlib.h> #include <string.h> #include <ctype.h> const char *base64_table = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; char *base64_encode(const unsigned char *data, size_t input_length, size_t *output_length) { *output_length = 4 * ((input_length + 2) / 3); char *encoded_data = malloc(*output_length + 1); if (encoded_data == NULL) return NULL; for (size_t i = 0, j = 0; i < input_length;) { uint32_t octet_a = i < input_length ? data[i++] : 0; uint32_t octet_b = i < input_length ? data[i++] : 0; uint32_t octet_c = i < input_length ? data[i++] : 0; uint32_t triple = (octet_a << 0x10) + (octet_b << 0x08) + octet_c; encoded_data[j++] = base64_table[(triple >> 3 * 6) & 0x3F]; encoded_data[j++] = base64_table[(triple >> 2 * 6) & 0x3F]; encoded_data[j++] = base64_table[(triple >> 1 * 6) & 0x3F]; encoded_data[j++] = base64_table[(triple >> 0 * 6) & 0x3F]; } for (size_t i = 0; i < (*output_length % 4); i++) encoded_data[*output_length - 1 - i] = '='; encoded_data[*output_length] = '\0'; return encoded_data; } 2. 解码 Base64解码过程就是将Base64编码的字符串转换为二进制数据。 解码的过程如下: - 将传入的字符串中的“=”字符去掉,将字符串长度调整为4的整数倍。 - 对于每4个字符,将其转换为3个6位的数字,即18位。 - 将这3个数字转换为3个字节的二进制数据。 - 如果编码的数据末尾有1或2个“=”字符,则表示解码后的数据末尾有1或2个字节为0。 C语言中可以使用标准库中的函数实现Base64解码,具体实现如下: #include <stdio.h> #include <stdlib.h> #include <string.h> #include <ctype.h> const char *base64_table = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; int base64_decode_char(char c) { if (isupper(c)) return c - 'A'; if (islower(c)) return c - 'a' + 26; if (isdigit(c)) return c - '0' + 52; if (c == '+') return 62; if (c == '/') return 63; return -1; } unsigned char *base64_decode(const char *data, size_t input_length, size_t *output_length) { if (input_length % 4 != 0) return NULL; *output_length = input_length / 4 * 3; if (data[input_length - 1] == '=') (*output_length)--; if (data[input_length - 2] == '=') (*output_length)--; unsigned char *decoded_data = malloc(*output_length); if (decoded_data == NULL) return NULL; for (size_t i = 0, j = 0; i < input_length;) { uint32_t sextet_a = base64_decode_char(data[i++]); uint32_t sextet_b = base64_decode_char(data[i++]); uint32_t sextet_c = base64_decode_char(data[i++]); uint32_t sextet_d = base64_decode_char(data[i++]); uint32_t triple = (sextet_a << 3 * 6) + (sextet_b << 2 * 6) + (sextet_c << 1 * 6) + (sextet_d << 0 * 6); if (j < *output_length) decoded_data[j++] = (triple >> 2 * 8) & 0xFF; if (j < *output_length) decoded_data[j++] = (triple >> 1 * 8) & 0xFF; if (j < *output_length) decoded_data[j++] = (triple >> 0 * 8) & 0xFF; } return decoded_data; } 以上就是Base64编码解码的C语言实现。使用时只需要调用相应的函数即可。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值