C/C++学习笔记 dlib中的base64编码源码分析

2024-03-06 10:59

本文主要是介绍C/C++学习笔记 dlib中的base64编码源码分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、头文件解析

        iostream标准输入输出流

        sstream 字符串输入输出流

        头文件climits定义了符号常量来表示类型的限制,编译器厂商提供了climits文件,该文件支持了其编译器中的值。例如,在使用16位int的老系统中,climits文件将INT_MAX定义为32676。

#include "base64_kernel_1.h"
#include <iostream>
#include <sstream>
#include <climits>

         fstream 文件输入输出流

二、换行符CR,LF和CRLF

        CR:\r 表示回车,MacIntosh操作系统(即早期的Mac操作系统)采用单个字符CR来进行换行

        LF:\n 转义字符表示换行,Unix/Linux/Mac OS X操作系统采用单个字符LF来进行换行

        CRLF:\r\n 表示回车并换行,Windows操作系统采用两个字符来进行换行,即CRLF

enum line_ending_type
{CR,  // i.e. "\r"LF,  // i.e. "\n"CRLF // i.e. "\r\n"
};// ----------------------------------------------------------------------------------------base64::line_ending_type base64::line_ending () const{return eol_style;}// ----------------------------------------------------------------------------------------void base64::set_line_ending (line_ending_type eol_style_){eol_style = eol_style_;}// ----------------------------------------------------------------------------------------

        根据RFC822规定,BASE64Encoder编码每76个字符,还需要加上一个回车换行,所以dlib这里定义了换行符。

三、构造函数

        这里组织了编码、解码的字符数组,下面是关于初始值的说明。

/*!INITIAL VALUE- bad_value == 100- encode_table == a pointer to an array of 64 chars- where x is a 6 bit value the following is true:- encode_table[x] == the base64 encoding of x- decode_table == a pointer to an array of UCHAR_MAX chars- where x is any char value:- if (x is a valid character in the base64 coding scheme) then- decode_table[x] == the 6 bit value that x encodes- else- decode_table[x] == bad_value CONVENTION- The state of this object never changes so just refer to itsinitial value.!*/

        源码代码

    base64::base64 () : encode_table(0),decode_table(0),bad_value(100),eol_style(LF){try{encode_table = new char[64];decode_table = new unsigned char[UCHAR_MAX];}catch (...){if (encode_table) delete [] encode_table;if (decode_table) delete [] decode_table;throw;}// now set up the tables with the right stuffencode_table[0] = 'A';encode_table[17] = 'R';encode_table[34] = 'i';encode_table[51] = 'z';encode_table[1] = 'B';encode_table[18] = 'S';encode_table[35] = 'j';encode_table[52] = '0';encode_table[2] = 'C';encode_table[19] = 'T';encode_table[36] = 'k';encode_table[53] = '1';encode_table[3] = 'D';encode_table[20] = 'U';encode_table[37] = 'l';encode_table[54] = '2';encode_table[4] = 'E';encode_table[21] = 'V';encode_table[38] = 'm';encode_table[55] = '3';encode_table[5] = 'F';encode_table[22] = 'W';encode_table[39] = 'n';encode_table[56] = '4';encode_table[6] = 'G';encode_table[23] = 'X';encode_table[40] = 'o';encode_table[57] = '5';encode_table[7] = 'H';encode_table[24] = 'Y';encode_table[41] = 'p';encode_table[58] = '6';encode_table[8] = 'I';encode_table[25] = 'Z';encode_table[42] = 'q';encode_table[59] = '7';encode_table[9] = 'J';encode_table[26] = 'a';encode_table[43] = 'r';encode_table[60] = '8';encode_table[10] = 'K';encode_table[27] = 'b';encode_table[44] = 's';encode_table[61] = '9';encode_table[11] = 'L';encode_table[28] = 'c';encode_table[45] = 't';encode_table[62] = '+';encode_table[12] = 'M';encode_table[29] = 'd';encode_table[46] = 'u';encode_table[63] = '/';encode_table[13] = 'N';encode_table[30] = 'e';encode_table[47] = 'v';encode_table[14] = 'O';encode_table[31] = 'f';encode_table[48] = 'w';encode_table[15] = 'P';encode_table[32] = 'g';encode_table[49] = 'x';encode_table[16] = 'Q';encode_table[33] = 'h';encode_table[50] = 'y';// we can now fill out the decode_table by using the encode_tablefor (int i = 0; i < UCHAR_MAX; ++i){decode_table[i] = bad_value;}for (unsigned char i = 0; i < 64; ++i){decode_table[(unsigned char)encode_table[i]] = i;}}

四、析构函数

        这里删除了构造函数里面的数组。

    base64::~base64 (){delete [] encode_table;delete [] decode_table;}

五、编码函数

        1、这里首先了解streambuf::sgetn,是用于从streambuf对象的缓冲区中获取字符序列,获取 get 指针后面的nCount个字符并将它们存储在从pch开始的区域中。当streambuf对象中剩余的字符少于nCount时, sgetn获取剩余的任何字符。该函数重新定位 get 指针以跟随获取的字符。

        2、每76个字符,还需要加上一个回车换行,所以看到counter = 19,这里的76个字符是指写入到输出流的。

    void base64::encode (std::istream& in_,std::ostream& out_) const{using namespace std;streambuf& in = *in_.rdbuf();streambuf& out = *out_.rdbuf();unsigned char inbuf[3];unsigned char outbuf[4];streamsize status = in.sgetn(reinterpret_cast<char*>(&inbuf),3);unsigned char c1, c2, c3, c4, c5, c6;int counter = 19;// while we haven't hit the end of the input streamwhile (status != 0){if (counter == 0){counter = 19;// write a newlinechar ch;switch (eol_style){case CR:ch = '\r';if (out.sputn(&ch,1)!=1)throw std::ios_base::failure("error occurred in the base64 object");break;case LF:ch = '\n';if (out.sputn(&ch,1)!=1)throw std::ios_base::failure("error occurred in the base64 object");break;case CRLF:ch = '\r';if (out.sputn(&ch,1)!=1)throw std::ios_base::failure("error occurred in the base64 object");ch = '\n';if (out.sputn(&ch,1)!=1)throw std::ios_base::failure("error occurred in the base64 object");break;default:DLIB_CASSERT(false,"this should never happen");}}--counter;if (status == 3){// encode the bytes in inbuf to base64 and write them to the output streamc1 = inbuf[0]&0xfc;c2 = inbuf[0]&0x03;c3 = inbuf[1]&0xf0;c4 = inbuf[1]&0x0f;c5 = inbuf[2]&0xc0;c6 = inbuf[2]&0x3f;outbuf[0] = c1>>2;outbuf[1] = (c2<<4)|(c3>>4);outbuf[2] = (c4<<2)|(c5>>6);outbuf[3] = c6;outbuf[0] = encode_table[outbuf[0]];outbuf[1] = encode_table[outbuf[1]];outbuf[2] = encode_table[outbuf[2]];outbuf[3] = encode_table[outbuf[3]];// write the encoded bytes to the output streamif (out.sputn(reinterpret_cast<char*>(&outbuf),4)!=4){throw std::ios_base::failure("error occurred in the base64 object");}// get 3 more input bytesstatus = in.sgetn(reinterpret_cast<char*>(&inbuf),3);continue;}else if (status == 2){// we are at the end of the input stream and need to add some padding// encode the bytes in inbuf to base64 and write them to the output streamc1 = inbuf[0]&0xfc;c2 = inbuf[0]&0x03;c3 = inbuf[1]&0xf0;c4 = inbuf[1]&0x0f;c5 = 0;outbuf[0] = c1>>2;outbuf[1] = (c2<<4)|(c3>>4);outbuf[2] = (c4<<2)|(c5>>6);outbuf[3] = '=';outbuf[0] = encode_table[outbuf[0]];outbuf[1] = encode_table[outbuf[1]];outbuf[2] = encode_table[outbuf[2]];// write the encoded bytes to the output streamif (out.sputn(reinterpret_cast<char*>(&outbuf),4)!=4){throw std::ios_base::failure("error occurred in the base64 object");}break;}else // in this case status must be 1 {// we are at the end of the input stream and need to add some padding// encode the bytes in inbuf to base64 and write them to the output streamc1 = inbuf[0]&0xfc;c2 = inbuf[0]&0x03;c3 = 0;outbuf[0] = c1>>2;outbuf[1] = (c2<<4)|(c3>>4);outbuf[2] = '=';outbuf[3] = '=';outbuf[0] = encode_table[outbuf[0]];outbuf[1] = encode_table[outbuf[1]];// write the encoded bytes to the output streamif (out.sputn(reinterpret_cast<char*>(&outbuf),4)!=4){throw std::ios_base::failure("error occurred in the base64 object");}break;}} // while (status != 0)// make sure the stream buffer flushes to its I/O channelout.pubsync();}

六、解码函数

    void base64::decode (std::istream& in_,std::ostream& out_) const{using namespace std;streambuf& in = *in_.rdbuf();streambuf& out = *out_.rdbuf();unsigned char inbuf[4];unsigned char outbuf[3];int inbuf_pos = 0;streamsize status = in.sgetn(reinterpret_cast<char*>(inbuf),1);// only count this character if it isn't some kind of fillerif (status == 1 && decode_table[inbuf[0]] != bad_value )++inbuf_pos;unsigned char c1, c2, c3, c4, c5, c6;streamsize outsize;// while we haven't hit the end of the input streamwhile (status != 0){// if we have 4 valid charactersif (inbuf_pos == 4){inbuf_pos = 0;// this might be the end of the encoded data so we need to figure out if // there was any padding applied.outsize = 3;if (inbuf[3] == '='){if (inbuf[2] == '=')outsize = 1;elseoutsize = 2;}// decode the incoming charactersinbuf[0] = decode_table[inbuf[0]];inbuf[1] = decode_table[inbuf[1]];inbuf[2] = decode_table[inbuf[2]];inbuf[3] = decode_table[inbuf[3]];// now pack these guys into bytes rather than 6 bit chunksc1 = inbuf[0]<<2;c2 = inbuf[1]>>4;c3 = inbuf[1]<<4;c4 = inbuf[2]>>2;c5 = inbuf[2]<<6;c6 = inbuf[3];outbuf[0] = c1|c2;outbuf[1] = c3|c4;outbuf[2] = c5|c6;// write the encoded bytes to the output streamif (out.sputn(reinterpret_cast<char*>(&outbuf),outsize)!=outsize){throw std::ios_base::failure("error occurred in the base64 object");}}// get more input characters status = in.sgetn(reinterpret_cast<char*>(inbuf + inbuf_pos),1);// only count this character if it isn't some kind of filler if ((decode_table[inbuf[inbuf_pos]] != bad_value || inbuf[inbuf_pos] == '=') && status != 0)++inbuf_pos;} // while (status != 0)if (inbuf_pos != 0){ostringstream sout;sout << inbuf_pos << " extra characters were found at the end of the encoded data."<< "  This may indicate that the data stream has been truncated.";// this happens if we hit EOF in the middle of decoding a 24bit block.throw decode_error(sout.str());}// make sure the stream buffer flushes to its I/O channelout.pubsync();}

七、源码地址

dlib/dlib/base64 at master · davisking/dlib · GitHubA toolkit for making real world machine learning and data analysis applications in C++ - dlib/dlib/base64 at master · davisking/dlibhttps://github.com/davisking/dlib/tree/master/dlib/base64

这篇关于C/C++学习笔记 dlib中的base64编码源码分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/779849

相关文章

Go标准库常见错误分析和解决办法

《Go标准库常见错误分析和解决办法》Go语言的标准库为开发者提供了丰富且高效的工具,涵盖了从网络编程到文件操作等各个方面,然而,标准库虽好,使用不当却可能适得其反,正所谓工欲善其事,必先利其器,本文将... 目录1. 使用了错误的time.Duration2. time.After导致的内存泄漏3. jsO

Python使用自带的base64库进行base64编码和解码

《Python使用自带的base64库进行base64编码和解码》在Python中,处理数据的编码和解码是数据传输和存储中非常普遍的需求,其中,Base64是一种常用的编码方案,本文我将详细介绍如何使... 目录引言使用python的base64库进行编码和解码编码函数解码函数Base64编码的应用场景注意

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

找不到Anaconda prompt终端的原因分析及解决方案

《找不到Anacondaprompt终端的原因分析及解决方案》因为anaconda还没有初始化,在安装anaconda的过程中,有一行是否要添加anaconda到菜单目录中,由于没有勾选,导致没有菜... 目录问题原因问http://www.chinasem.cn题解决安装了 Anaconda 却找不到 An

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque

C++常见容器获取头元素的方法大全

《C++常见容器获取头元素的方法大全》在C++编程中,容器是存储和管理数据集合的重要工具,不同的容器提供了不同的接口来访问和操作其中的元素,获取容器的头元素(即第一个元素)是常见的操作之一,本文将详细... 目录一、std::vector二、std::list三、std::deque四、std::forwa