BASE64编码和解码(VC源代码) 并 内存加载 CImage 图像

转自:http://www.cnblogs.com/lujin49/p/4957742.html

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

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

BASE64_API.h  文件内容

 1 /* ----------------------------------------------------------
 2 文件名称:BASE64_API.h
 3 
 4 作者:秦建辉
 5 
 6 MSN:splashcn@msn.com
 7 
 8 当前版本:V1.1
 9 
10 历史版本:
11     V1.1    2010年05月11日
12             修正BASE64解码的Bug。
13 
14     V1.0    2010年05月07日
15             完成正式版本。
16 
17 功能描述:
18     BASE64编码和解码
19 
20 接口函数:
21     Base64_Encode
22     Base64_Decode
23 
24 说明:
25     1.    参考openssl-1.0.0。
26     2.    改进接口,以使其适应TCHAR字符串。
27     3.    修正EVP_DecodeBlock函数解码时未去掉填充字节的缺陷。
28  ------------------------------------------------------------ */
29 #pragma once
30 #include "stdafx.h"#include <windows.h>
31 
32 #ifdef    __cplusplus
33 extern "C" {
34 #endif
35 
36 /*
37 功能:将二进制数据转换成BASE64编码字符串
38 参数说明:
39     inputBuffer:要编码的二进制数据
40     inputCount:数据长度
41     outputBuffer:存储转换后的BASE64编码字符串
42 返回值:
43      -1:参数错误
44     >=0:有效编码长度(字符数),不包括字符串结束符。
45 备注:
46     等效于openssl中EVP_EncodeBlock函数
47 */
48 INT BASE64_Encode( const BYTE* inputBuffer, INT inputCount, TCHAR* outputBuffer );
49 
50 /*
51 功能:将BASE64编码字符串转换为二进制数据
52 参数说明:
53     inputBuffer:BASE64编码字符串
54     inputCount:编码长度(字符数),应该为4的倍数。
55     outputBuffer:存储转换后的二进制数据
56 返回值:
57      -1:参数错误
58      -2:数据错误
59     >=0:转换后的字节数
60 备注:
61     等效于openssl中EVP_DecodeBlock函数
62 */
63 INT BASE64_Decode( const TCHAR* inputBuffer, INT inputCount, BYTE* outputBuffer );
64 
65 #ifdef __cplusplus
66 }
67 #endif

BASE64_API.cpp 文件内容

  1 #pragma once
  2 #include "stdafx.h"#include "BASE64_API.h"
  3 
  4 static const CHAR* DATA_BIN2ASCII = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
  5 
  6 INT BASE64_Encode( const BYTE* inputBuffer, INT inputCount, TCHAR* outputBuffer )
  7 {
  8     INT i;
  9     BYTE b0, b1, b2;
 10 
 11     if( (inputBuffer == NULL) || (inputCount < 0) )
 12     {
 13         return -1;    // 参数错误
 14     }
 15 
 16     if( outputBuffer != NULL )
 17     {
 18         for( i = inputCount; i > 0; i -= 3 )
 19         {
 20             if( i >= 3 )
 21             {    // 将3字节数据转换成4个ASCII字符
 22                 b0 = *inputBuffer++;
 23                 b1 = *inputBuffer++;
 24                 b2 = *inputBuffer++;
 25 
 26                 *outputBuffer++ = DATA_BIN2ASCII[b0 >> 2];
 27                 *outputBuffer++ = DATA_BIN2ASCII[((b0 << 4) | (b1 >> 4)) & 0x3F];
 28                 *outputBuffer++ = DATA_BIN2ASCII[((b1 << 2) | (b2 >> 6)) & 0x3F];
 29                 *outputBuffer++ = DATA_BIN2ASCII[b2 & 0x3F];
 30             }
 31             else
 32             {
 33                 b0 = *inputBuffer++;
 34                 if( i == 2 )b1 = *inputBuffer++; else b1 = 0;
 35 
 36                 *outputBuffer++ = DATA_BIN2ASCII[b0 >> 2];
 37                 *outputBuffer++ = DATA_BIN2ASCII[((b0 << 4) | (b1 >> 4)) & 0x3F];
 38                 *outputBuffer++ = (i == 1) ? TEXT('=') : DATA_BIN2ASCII[(b1 << 2) & 0x3F];
 39                 *outputBuffer++ = TEXT('=');
 40             }
 41         } // End for i
 42 
 43         *outputBuffer++ = TEXT('/0');    // 添加字符串结束标记
 44     }
 45 
 46     return ((inputCount + 2) / 3) * 4;    // 返回有效字符个数
 47 }
 48 
 49 #define B64_EOLN            0xF0    // 换行/n
 50 #define B64_CR                0xF1    // 回车/r
 51 #define B64_EOF                0xF2    // 连字符-
 52 #define B64_WS                0xE0    // 跳格或者空格(/t、space)
 53 #define B64_ERROR     0xFF    // 错误字符
 54 #define B64_NOT_BASE64(a)    (((a)|0x13) == 0xF3)
 55 
 56 static const BYTE DATA_ASCII2BIN[128] = {
 57     0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xE0,0xF0,0xFF,0xFF,0xF1,0xFF,0xFF,
 58     0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,
 59     0xE0,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0x3E,0xFF,0xF2,0xFF,0x3F,
 60     0x34,0x35,0x36,0x37,0x38,0x39,0x3A,0x3B,0x3C,0x3D,0xFF,0xFF,0xFF,0x00,0xFF,0xFF,
 61     0xFF,0x00,0x01,0x02,0x03,0x04,0x05,0x06,0x07,0x08,0x09,0x0A,0x0B,0x0C,0x0D,0x0E,
 62     0x0F,0x10,0x11,0x12,0x13,0x14,0x15,0x16,0x17,0x18,0x19,0xFF,0xFF,0xFF,0xFF,0xFF,
 63     0xFF,0x1A,0x1B,0x1C,0x1D,0x1E,0x1F,0x20,0x21,0x22,0x23,0x24,0x25,0x26,0x27,0x28,
 64     0x29,0x2A,0x2B,0x2C,0x2D,0x2E,0x2F,0x30,0x31,0x32,0x33,0xFF,0xFF,0xFF,0xFF,0xFF
 65 };
 66 
 67 INT BASE64_Decode( const TCHAR* inputBuffer, INT inputCount, BYTE* outputBuffer )
 68 {
 69     INT i, j;
 70     BYTE b[4];
 71     TCHAR ch;
 72 
 73     if( (inputBuffer == NULL) || (inputCount < 0) )
 74     {
 75         return -1;    // 参数错误
 76     }
 77 
 78     // 去除头部空白字符
 79     while( inputCount > 0 )
 80     {
 81         ch = *inputBuffer;
 82         if( (ch < 0) || (ch >= 0x80) )
 83         {
 84             return -2;    // 数据错误,不在ASCII字符编码范围内
 85         }
 86         else
 87         {
 88             if( DATA_ASCII2BIN[ch] == B64_WS )
 89             {
 90                 inputBuffer++;
 91                 inputCount--;
 92             }
 93             else
 94             {
 95                 break;
 96             }
 97         }
 98     }
 99 
100     // 去除尾部的空白字符、回车换行字符、连字符
101     while( inputCount >= 4 )
102     {
103         ch = inputBuffer[inputCount - 1];
104         if( (ch < 0) || (ch >= 0x80) )
105         {
106             return -2;    // 数据错误,不在ASCII字符编码范围内
107         }
108         else
109         {
110             if( B64_NOT_BASE64(DATA_ASCII2BIN[ch]) )
111             {
112                 inputCount--;
113             }
114             else
115             {
116                 break;
117             }
118         }
119     }
120 
121     // 字符串长度必须为4的倍数
122     if( (inputCount % 4) != 0 )
123     {
124         return -2;    // 数据错误
125     }
126 
127     if( outputBuffer != NULL )
128     {
129         for( i = 0; i < inputCount; i += 4 )
130         {
131             for( j = 0; j < 4; j++ )
132             {
133                 ch = *inputBuffer++;
134                 if( (ch < 0) || (ch >= 0x80) )
135                 {
136                     return -2;    // 数据错误,不在ASCII字符编码范围内
137                 }
138                 else
139                 {
140                     if( ch == '=' )    // 发现BASE64编码中的填充字符
141                     {
142                         break;
143                     }
144                     else
145                     {
146                         b[j] = DATA_ASCII2BIN[ch];
147                         if( b[j] & 0x80 )
148                         {
149                             return -2;    // 数据错误,无效的Base64编码字符
150                         }
151                     }                    
152                 }
153             } // End for j
154 
155             if( j == 4 )
156             {
157                 *outputBuffer++ = (b[0] << 2) | (b[1] >> 4);
158                 *outputBuffer++ = (b[1] << 4) | (b[2] >> 2 );
159                 *outputBuffer++ = (b[2] << 6) | b[3];
160             }
161             else if( j == 3 )
162             {    // 有1个填充字节
163                 *outputBuffer++ = (b[0] << 2) | (b[1] >> 4);
164                 *outputBuffer++ = (b[1] << 4) | (b[2] >> 2 );
165 
166                 return (i >> 2) * 3 + 2;
167             }
168             else if( j == 2 )
169             {    // 有2个填充字节
170                 *outputBuffer++ = (b[0] << 2) | (b[1] >> 4);
171 
172                 return (i >> 2) * 3 + 1;
173             }
174             else
175             {
176                 return -2;    // 数据错误,无效的Base64编码字符
177             }            
178         }    // End for i
179     }
180 
181     return (inputCount >> 2) * 3;
182 }

采用以上方法就可以将二进制数据转换成可见字符进行传递就可以了.

那么如何使用呢?举以下两个例子

第一个:将一个图片转换成 txt 文本 并保存起来

 1 //选择一个图像文件,将它转为 文本保存至  _T("D:\2.txt"
 2 void CTextPicDlg::OnBnClickedButton2()
 3 {
 4     // TODO:  在此添加控件通知处理程序代码
 5 
 6     CFileDialog file(TRUE,".jpg","");
 7     if (file.DoModal() == IDOK)
 8     {
 9         CFile data(file.GetPathName(), CFile::modeReadWrite);
10         int len = data.GetLength();
11         BYTE *dv;
12         dv = (BYTE *)malloc(len*sizeof(BYTE));
13         data.Read(dv, len);
14         data.Close();
15 
16         int slen = (len / 3) * 4;
17         slen += 10;
18         TCHAR * tc;
19         tc = (TCHAR *)malloc(slen);
20 
21         slen = BASE64_Encode(dv, len, tc);
22 
23         CFile save(_T("D:\2.txt"), CFile::modeCreate | CFile::modeWrite);
24         save.Write(tc, slen);
25         save.Close();
26 
27         free(tc);
28         free(dv);
29         
30     }
31 
32 }

第二个例子,将一个文本文件还原为一个图像

 1 void CTextPicDlg::OnBnClickedButton3()
 2 {
 3     // TODO:  在此添加控件通知处理程序代码
 4 
 5     CFileDialog file(TRUE, ".txt", "");
 6     if (file.DoModal() == IDOK)
 7     {
 8         CFile data(file.GetPathName(), CFile::modeReadWrite);
 9         int len = data.GetLength();
10         TCHAR *dv;
11         dv = (TCHAR *)malloc(len*sizeof(TCHAR));
12         data.Read(dv, len);
13         data.Close();
14 
15         int slen = (len / 4) * 3;
16         slen += 10;
17         BYTE * tc;
18         tc = (BYTE *)malloc(slen);
19 
20         BASE64_Decode(dv, len, tc);
21          //直接在内存里面构建CIMAGE,需要使用IStream接口,如何使用        //构建内存环境        
22         HGLOBAL hGlobal = GlobalAlloc(GMEM_MOVEABLE, slen);
23         void *  pData = GlobalLock(hGlobal);
24         memcpy(pData, tc, slen);  //  拷贝位图数据进去
25         GlobalUnlock(hGlobal);
26 
27         //  创建IStream
28         IStream *  pStream = NULL;
29         if (CreateStreamOnHGlobal(hGlobal, TRUE, &pStream) != S_OK)
30             return ;
31         //   使用CImage加载位图内存
32         CImage img;
33         if (SUCCEEDED(img.Load(pStream)) )
34         {
35             CClientDC dc(this);
36 
37               //使用内在中构造的图像 直接在对话框上绘图
38              img.Draw(dc.m_hDC, 0, 0, 500, 300);
39 
40 
41             
42         }
43          //释放内存
44         pStream->Release();
45         GlobalFree(hGlobal);
46 
47 
48         //如果要保存图像文件的话,那就使用下面的代码
49         //CFileDialog savefile(FALSE, ".jpg", "");
50         //if (savefile.DoModal()==IDOK)
51         //{
52         //    CFile save(savefile.GetPathName(), CFile::modeCreate | CFile::modeWrite);
53         //    save.Write(tc, slen);
54         //    save.Close();
55         //}
56 
57         free(tc);
58         free(dv);
59 
60     }
61 
62 }

至此,利用Base64转码的方式,来显示保存显示图片的方法,就算是成功了!

原文地址:https://www.cnblogs.com/lpxblog/p/5330667.html