Chinaunix首页 | 论坛 | 博客
  • 博客访问: 640416
  • 博文数量: 133
  • 博客积分: 1566
  • 博客等级: 上尉
  • 技术积分: 1230
  • 用 户 组: 普通用户
  • 注册时间: 2010-12-01 09:31
文章分类

全部博文(133)

文章存档

2019年(1)

2018年(1)

2017年(8)

2016年(9)

2015年(17)

2014年(4)

2013年(31)

2012年(25)

2011年(36)

2010年(1)

我的朋友

分类: C/C++

2013-06-18 09:36:35

    BASE64可以用来将binary的字节序列数据编码成ASCII字符序列构成的文本。完整的BASE64定义可见 RFC1421和 RFC2045。编码后的数据比原始数据略长,为原来的4/3。在电子邮件中,根据RFC822规定,每76个字符,还需要加上一个回车换行。

    转换的时候,将三个byte的数据,先后放入一个24bit的缓冲区中,先来的byte占高位。数据不足3byte的话,于缓冲区中剩下的Bit用0补足。然后,每次取出6个bit,按照其值选择ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/中的字符作为编码后的输出。不断进行,直到全部输入数据转换完成。如果最后剩下两个输入数据,在编码结果后加1个“=”;如果最后剩下一个输入数据,编码结果后加2个“=”;如果没有剩下任何数据,就什么都不要加,这样才可以保证资料还原的正确性。

 

开发语言:C/C++

实现功能:实现BASE64编码和解码,提供如下接口函数:

BASE64_Encode
BASE64_Decode

下载地址:

更新历史:

V1.1 2010年05月11日

  • 修正BASE64解码的Bug。

V1.0 2010年05月07日

  • 完成正式版本。

附加说明:

  1. 参考openssl-1.0.0。
  2. 改进接口,以使其适应TCHAR字符串。
  3. 修正EVP_DecodeBlock函数解码时未去掉填充字节的缺陷。

BASE64_API.h:

点击(此处)折叠或打开

  1. /* ----------------------------------------------------------
  2. 文件名称:BASE64_API.h

  3. 作者:秦建辉

  4. MSN:splashcn@msn.com

  5. 当前版本:V1.1

  6. 历史版本:
  7.     V1.1    2010年05月11日
  8.             修正BASE64解码的Bug。

  9.     V1.0    2010年05月07日
  10.             完成正式版本。

  11. 功能描述:
  12.     BASE64编码和解码

  13. 接口函数:
  14.     Base64_Encode
  15.     Base64_Decode

  16. 说明:
  17.     1.    参考openssl-1.0.0。
  18.     2.    改进接口,以使其适应TCHAR字符串。
  19.     3.    修正EVP_DecodeBlock函数解码时未去掉填充字节的缺陷。
  20.  ------------------------------------------------------------ */
  21. #pragma once

  22. #include <windows.h>

  23. #ifdef    __cplusplus
  24. extern "C" {
  25. #endif

  26. /*
  27. 功能:将二进制数据转换成BASE64编码字符串
  28. 参数说明:
  29.     inputBuffer:要编码的二进制数据
  30.     inputCount:数据长度
  31.     outputBuffer:存储转换后的BASE64编码字符串
  32. 返回值:
  33.      -1:参数错误
  34.     >=0:有效编码长度(字符数),不包括字符串结束符。
  35. 备注:
  36.     等效于openssl中EVP_EncodeBlock函数
  37. */
  38. INT BASE64_Encode( const BYTE* inputBuffer, INT inputCount, TCHAR* outputBuffer );

  39. /*
  40. 功能:将BASE64编码字符串转换为二进制数据
  41. 参数说明:
  42.     inputBuffer:BASE64编码字符串
  43.     inputCount:编码长度(字符数),应该为4的倍数。
  44.     outputBuffer:存储转换后的二进制数据
  45. 返回值:
  46.      -1:参数错误
  47.      -2:数据错误
  48.     >=0:转换后的字节数
  49. 备注:
  50.     等效于openssl中EVP_DecodeBlock函数
  51. */
  52. INT BASE64_Decode( const TCHAR* inputBuffer, INT inputCount, BYTE* outputBuffer );

  53. #ifdef __cplusplus
  54. }
  55. #endif


BASE64_API.cpp:

点击(此处)折叠或打开

  1. #include "BASE64_API.h"

  2. static const CHAR* DATA_BIN2ASCII = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

  3. INT BASE64_Encode( const BYTE* inputBuffer, INT inputCount, TCHAR* outputBuffer )
  4. {
  5.     INT i;
  6.     BYTE b0, b1, b2;

  7.     if( (inputBuffer == NULL) || (inputCount < 0) )
  8.     {
  9.         return -1;    // 参数错误
  10.     }

  11.     if( outputBuffer != NULL )
  12.     {
  13.         for( i = inputCount; i > 0; i -= 3 )
  14.         {
  15.             if( i >= 3 )
  16.             {    // 将3字节数据转换成4个ASCII字符
  17.                 b0 = *inputBuffer++;
  18.                 b1 = *inputBuffer++;
  19.                 b2 = *inputBuffer++;

  20.                 *outputBuffer++ = DATA_BIN2ASCII[b0 >> 2];
  21.                 *outputBuffer++ = DATA_BIN2ASCII[((b0 << 4) | (b1 >> 4)) & 0x3F];
  22.                 *outputBuffer++ = DATA_BIN2ASCII[((b1 << 2) | (b2 >> 6)) & 0x3F];
  23.                 *outputBuffer++ = DATA_BIN2ASCII[b2 & 0x3F];
  24.             }
  25.             else
  26.             {
  27.                 b0 = *inputBuffer++;
  28.                 if( i == 2 )b1 = *inputBuffer++; else b1 = 0;

  29.                 *outputBuffer++ = DATA_BIN2ASCII[b0 >> 2];
  30.                 *outputBuffer++ = DATA_BIN2ASCII[((b0 << 4) | (b1 >> 4)) & 0x3F];
  31.                 *outputBuffer++ = (i == 1) ? TEXT('=') : DATA_BIN2ASCII[(b1 << 2) & 0x3F];
  32.                 *outputBuffer++ = TEXT('=');
  33.             }
  34.         } // End for i

  35.         *outputBuffer++ = TEXT('/0');    // 添加字符串结束标记
  36.     }

  37.     return ((inputCount + 2) / 3) * 4;    // 返回有效字符个数
  38. }

  39. #define B64_EOLN            0xF0    // 换行/n
  40. #define B64_CR                0xF1    // 回车/r
  41. #define B64_EOF                0xF2    // 连字符-
  42. #define B64_WS                0xE0    // 跳格或者空格(/t、space)
  43. #define B64_ERROR     0xFF    // 错误字符
  44. #define B64_NOT_BASE64(a)    (((a)|0x13) == 0xF3)

  45. static const BYTE DATA_ASCII2BIN[128] = {
  46.     0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xE0,0xF0,0xFF,0xFF,0xF1,0xFF,0xFF,
  47.     0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,
  48.     0xE0,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0x3E,0xFF,0xF2,0xFF,0x3F,
  49.     0x34,0x35,0x36,0x37,0x38,0x39,0x3A,0x3B,0x3C,0x3D,0xFF,0xFF,0xFF,0x00,0xFF,0xFF,
  50.     0xFF,0x00,0x01,0x02,0x03,0x04,0x05,0x06,0x07,0x08,0x09,0x0A,0x0B,0x0C,0x0D,0x0E,
  51.     0x0F,0x10,0x11,0x12,0x13,0x14,0x15,0x16,0x17,0x18,0x19,0xFF,0xFF,0xFF,0xFF,0xFF,
  52.     0xFF,0x1A,0x1B,0x1C,0x1D,0x1E,0x1F,0x20,0x21,0x22,0x23,0x24,0x25,0x26,0x27,0x28,
  53.     0x29,0x2A,0x2B,0x2C,0x2D,0x2E,0x2F,0x30,0x31,0x32,0x33,0xFF,0xFF,0xFF,0xFF,0xFF
  54. };

  55. INT BASE64_Decode( const TCHAR* inputBuffer, INT inputCount, BYTE* outputBuffer )
  56. {
  57.     INT i, j;
  58.     BYTE b[4];
  59.     TCHAR ch;

  60.     if( (inputBuffer == NULL) || (inputCount < 0) )
  61.     {
  62.         return -1;    // 参数错误
  63.     }

  64.     // 去除头部空白字符
  65.     while( inputCount > 0 )
  66.     {
  67.         ch = *inputBuffer;
  68.         if( (ch < 0) || (ch >= 0x80) )
  69.         {
  70.             return -2;    // 数据错误,不在ASCII字符编码范围内
  71.         }
  72.         else
  73.         {
  74.             if( DATA_ASCII2BIN[ch] == B64_WS )
  75.             {
  76.                 inputBuffer++;
  77.                 inputCount--;
  78.             }
  79.             else
  80.             {
  81.                 break;
  82.             }
  83.         }
  84.     }

  85.     // 去除尾部的空白字符、回车换行字符、连字符
  86.     while( inputCount >= 4 )
  87.     {
  88.         ch = inputBuffer[inputCount - 1];
  89.         if( (ch < 0) || (ch >= 0x80) )
  90.         {
  91.             return -2;    // 数据错误,不在ASCII字符编码范围内
  92.         }
  93.         else
  94.         {
  95.             if( B64_NOT_BASE64(DATA_ASCII2BIN[ch]) )
  96.             {
  97.                 inputCount--;
  98.             }
  99.             else
  100.             {
  101.                 break;
  102.             }
  103.         }
  104.     }

  105.     // 字符串长度必须为4的倍数
  106.     if( (inputCount % 4) != 0 )
  107.     {
  108.         return -2;    // 数据错误
  109.     }

  110.     if( outputBuffer != NULL )
  111.     {
  112.         for( i = 0; i < inputCount; i += 4 )
  113.         {
  114.             for( j = 0; j < 4; j++ )
  115.             {
  116.                 ch = *inputBuffer++;
  117.                 if( (ch < 0) || (ch >= 0x80) )
  118.                 {
  119.                     return -2;    // 数据错误,不在ASCII字符编码范围内
  120.                 }
  121.                 else
  122.                 {
  123.                     if( ch == '=' )    // 发现BASE64编码中的填充字符
  124.                     {
  125.                         break;
  126.                     }
  127.                     else
  128.                     {
  129.                         b[j] = DATA_ASCII2BIN[ch];
  130.                         if( b[j] & 0x80 )
  131.                         {
  132.                             return -2;    // 数据错误,无效的Base64编码字符
  133.                         }
  134.                     }                    
  135.                 }
  136.             } // End for j

  137.             if( j == 4 )
  138.             {
  139.                 *outputBuffer++ = (b[0] << 2) | (b[1] >> 4);
  140.                 *outputBuffer++ = (b[1] << 4) | (b[2] >> 2 );
  141.                 *outputBuffer++ = (b[2] << 6) | b[3];
  142.             }
  143.             else if( j == 3 )
  144.             {    // 有1个填充字节
  145.                 *outputBuffer++ = (b[0] << 2) | (b[1] >> 4);
  146.                 *outputBuffer++ = (b[1] << 4) | (b[2] >> 2 );

  147.                 return (i >> 2) * 3 + 2;
  148.             }
  149.             else if( j == 2 )
  150.             {    // 有2个填充字节
  151.                 *outputBuffer++ = (b[0] << 2) | (b[1] >> 4);

  152.                 return (i >> 2) * 3 + 1;
  153.             }
  154.             else
  155.             {
  156.                 return -2;    // 数据错误,无效的Base64编码字符
  157.             }            
  158.         }    // End for i
  159.     }

  160.     return (inputCount >> 2) * 3;
  161. }

转载:http://blog.csdn.net/jhqin/article/details/5568862


阅读(10112) | 评论(0) | 转发(2) |
给主人留下些什么吧!~~