• linkedu视频
  • 平面设计
  • 电脑入门
  • 操作系统
  • 办公应用
  • 电脑硬件
  • 动画设计
  • 3D设计
  • 网页设计
  • CAD设计
  • 影音处理
  • 数据库
  • 程序设计
  • 认证考试
  • 信息管理
  • 信息安全
菜单
linkedu.com
  • 网页制作
  • 数据库
  • 程序设计
  • 操作系统
  • CMS教程
  • 游戏攻略
  • 脚本语言
  • 平面设计
  • 软件教程
  • 网络安全
  • 电脑知识
  • 服务器
  • 视频教程
  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号
您的位置:首页 > 程序设计 >C语言 > VC中BASE64编码和解码使用详解

VC中BASE64编码和解码使用详解

作者: 字体:[增加 减小] 来源:互联网 时间:2017-05-28

通过本文主要向大家介绍了vc base64,base64编码详解,base64详解,base64编码,图片base64编码等相关知识,希望对您有所帮助,也希望大家支持linkedu.com www.linkedu.com

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

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

BASE64_API.h  文件内容

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

作者:秦建辉

MSN:splashcn@msn.com

当前版本:V1.1

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

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

功能描述:
  BASE64编码和解码

接口函数:
  Base64_Encode
  Base64_Decode

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

#include <windows.h>

#ifdef  __cplusplus
extern "C" {
#endif

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

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

#ifdef __cplusplus
}
#endif
</div>

BASE64_API.cpp 文件内容

#pragma once

#include "stdafx.h"

#include "BASE64_API.h"

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

INT BASE64_Encode( const BYTE* inputBuffer, INT inputCount, TCHAR* outputBuffer )
{
  INT i;
  BYTE b0, b1, b2;

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

  if( outputBuffer != NULL )
  {
    for( i = inputCount; i > 0; i -= 3 )
    {
      if( i >= 3 )
      {  // 将3字节数据转换成4个ASCII字符
        b0 = *inputBuffer++;
        b1 = *inputBuffer++;
        b2 = *inputBuffer++;

        *outputBuffer++ = DATA_BIN2ASCII[b0 >> 2];
        *outputBuffer++ = DATA_BIN2ASCII[((b0 << 4) | (b1 >> 4)) & 0x3F];
        *outputBuffer++ = DATA_BIN2ASCII[((b1 << 2) | (b2 >> 6)) & 0x3F];
        *outputBuffer++ = DATA_BIN2ASCII[b2 & 0x3F];
      }
      else
      {
        b0 = *inputBuffer++;
        if( i == 2 )b1 = *inputBuffer++; else b1 = 0;

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

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

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

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

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

INT BASE64_Decode( const TCHAR* inputBuffer, INT inputCount, BYTE* outputBuffer )
{
  INT i, j;
  BYTE b[4];
  TCHAR ch;

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

  // 去除头部空白字符
  while( inputCount > 0 )
  {
    ch = *inputBuffer;
    if( (ch < 0) || (ch >= 0x80) )
    {
      return -2;  // 数据错误,不在ASCII字符编码范围内
    }
    else
    {
      if( DATA_ASCII2BIN[ch] == B64_WS )
      {
        inputBuffer++;
        inputCount--;
      }
      else
      {
        break;
      }
    }
  }

  // 去除尾部的空白字符、回车换行字符、连字符
  while( inputCount >= 4 )
  {
    ch = inputBuffer[inputCount - 1];
    if( (ch < 0) || (ch >= 0x80) )
    {
      return -2;  // 数据错误,不在ASCII字符编码范围内
    }
    else
    {
      if( B64_NOT_BASE64(DATA_ASCII2BIN[ch]) )
      {
        inputCount--;
      }
      else
      {
        break;
      }
    }
  }

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

  if( outputBuffer != NULL )
  {
    for( i = 0; i < inputCount; i += 4 )
    {
      for( j = 0; j < 4; j++ )
      {
        ch = *inputBuffer++;
        if( (ch < 0) || (ch >= 0x80) )
        {
          return -2;  // 数据错误,不在ASCII字符编码范围内
        }
        else
        {
          if( ch == '=' )  // 发现BASE64编码中的填充字符
          {
            break;
          }
          else
          {
            b[j] = DATA_ASCII2BIN[ch];
            if( b[j] & 0x80 )
            {
              return -2;  // 数据错误,无效的Base64编码字符
            }
          }          
        }
      } // End for j

      if( j == 4 )
      {
        *outputBuffer++ = (b[0] << 2) | (b[1] >> 4);
        *outputBuffer++ = (b[1] << 4) | (b[2] >> 2 );
        *outputBuffer++ = (b[2] << 6) | b[3];
      }
      else if( j == 3 )
      {  // 有1个填充字节
        *outputBuffer++ = (b[0] << 2) | (b[1] >> 4);
        *outputBuffer++ = (b[1] << 4) | (b[2] >> 2 );

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

        return (i >> 2) * 3 + 1;
      }
      else
      {
        return -2;  // 数据错误,无效的Base64编码字符
      }      
    }  // End for i
  }

  return (inputCount >> 2) * 3;
}
</div>

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

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

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

//选择一个图像文件,将它转为 文本保存至 _T("D:\\2.txt"
void CTextPicDlg::OnBnClickedButton2()
{
  // TODO: 在此添加控件通知处理程序代码
  CFileDialog file(TRUE,".jpg","");
  if (file.DoModal() == IDOK)
  {
    CFile data(file.GetPathName(), CFile::modeReadWrite);
    int len = data.GetLength();
    BYTE *dv;
    dv = (BYTE *)malloc(len*sizeof(BYTE));




 
分享到:QQ空间新浪微博腾讯微博微信百度贴吧QQ好友复制网址打印

您可能想查找下面的文章:

  • VC中BASE64编码和解码使用详解

相关文章

  • 2017-05-28浅析C语言中strtol()函数与strtoul()函数的用法
  • 2017-05-28弦图ZOJ 1015 Fishing Net 判定方法
  • 2017-05-28C++读取INI配置文件类实例详解
  • 2017-05-28c语言 字符串转大写的简单实例
  • 2017-05-28贪吃蛇游戏C++命令行版实例代码
  • 2017-05-28详解C语言中的fopen()函数和fdopen()函数
  • 2017-05-28数据结构 数组顺序存储详细介绍
  • 2017-05-28深入VC回调函数的使用详解
  • 2017-05-28平衡二叉树的实现实例
  • 2017-05-28C语言实现颠倒栈的方法

文章分类

  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号

最近更新的内容

    • C++编程中用put输出单个字符和cin输入流的用法
    • C++中基类和派生类之间的转换实例教程
    • 解析C语言中如何正确使用const
    • C语言中对文件最基本的读取和写入函数
    • 素数筛选法
    • 实例讲解C++编程中对设计模式中的原型模式的使用
    • 基于C语言实现的aes256加密算法示例
    • C++ 学习之旅 Windows程序内部运行原理
    • 简单对比C语言中的fputs()函数和fputc()函数
    • HDOJ 1443 约瑟夫环的最新应用分析详解

关于我们 - 联系我们 - 免责声明 - 网站地图

©2020-2025 All Rights Reserved. linkedu.com 版权所有