Base64编码的C语言实现

本文转自http://blog.163.com/lixiangqiu_9202/blog/static/5357503720140593058179/

Bse64是一种以64个可打印字符对二进制数据进行编码的编码算法。base64在对数据进行编码时以三个8位字符型数据为一组,取这三个字符型数据的ASCII码,然后以6位为一组组成4个新的数据,这4个新的数据有6位,所以它的最大值为2^6=64。我们以4个6位数据的十进制数从base64表中得到最终编码后的字符。

Base64 编码

ValueChar ValueChar ValueChar ValueChar 
0A16Q32g48w
1B17R33h49x
2C18S34i50y
3D19T35j51z
4E20U36k520
5F21V37l531
6G22W38m542
7H23X39n553
8I24Y40o564
9J25Z41p575
10K26a42q586
11L27b43r597
12M28c44s608
13N29d45t619
14O30e46u62+
15P31f47v63/

       由于base64编码是将编码前的3*8位数据,分解成4个6位的数据,所以经过base64编码后的字符串长度是4的倍数。但往往我们进行编码的数据长度并不是3的倍数,这就造成了“编码”后的位数不为4的倍数,比如Brisk共5×8=40位,以6位为一组可以分为7组,这样“编码”后就有7个字符,但base64编码后的字符长度应该是4的倍数,显然这里就出问题了,那么怎么办呢?前面的不可以抛弃掉,所以就只有“追加”了,所以Brisk经过base64编码后的长度应该是8个字符,而第8个编码后的字符是'=',再比如对单个字符a进行base64编码,由于它的长度不是3的倍数,以3个字节为一组它只能分一组,再以6位为一位它只能分两组,所以经过“编码”后它的长度是2,但base64编码后的个数应该是4的倍数,所以它的长度应该是4,所以在后面补上两个‘=’,由于一个数求余3后有三个不同的结果,0、1、2,所以在对一个数据进行base64进行编码后它的长度为:

(1)当进行编码的数据长度是3的倍数时,len=strlen(str_in)/3*4;
(2)当进行编码的数据长度不是3的倍数时,len=(strlen(str_in)/3+1)*4;

       我们以Brisk这个例子来说明一下base64编码的过程。首先我们以3个字符为一组将Brisk进行分组Brisk被氛围两组:Bri 和 sk;然后我们取出这两个分组中每个字节的ASCII码,B:66 r:114 i:105 s:115 k:107。它们对应的二进制数为  B:01000010 r:01110010 i:01101001 s:01110011 k:01101011;

       第一组,我们以6位为一组对每一个3字节分组进行再分组就变成了010000 100111 001001 101001。所对应的十进制数是16 39 9 41,对应base64表中的结果是 Q n J p;

       第二组,011100 110110 101100(不够补0),所以对应的十进制数是 28 54 44,对应base64表中的结果是 c 2 s,最终结果为QnJpc2s=(因为第二组“编码”后只有三个字节)。

      解码的过程是一个逆过程,我们将经过编码后的字符按4个字符为一组,然后对照base64表得到相应的十进制数,再将其通过拆分和组合,组成3个8位数据,这个数据就是解码后的数据,下面给一个c语言实现编码和解码的代码。

[objc]  view plain  copy
  1. /*base64.h*/  
  2. #ifndef _BASE64_H  
  3. #define _BASE64_H  
  4.   
  5. #include <stdlib.h>  
  6. #include <string.h>  
  7.   
  8. unsigned charchar *base64_encode(unsigned charchar *str);  
  9.   
  10. unsigned charchar *bae64_decode(unsigned charchar *code);  
  11.   
  12. #endif  
  13.   
  14. /*base64.c*/  
  15. #include "base64.h"  
  16.   
  17. unsigned charchar *base64_encode(unsigned charchar *str)  
  18. {  
  19.     long len;  
  20.     long str_len;  
  21.     unsigned charchar *res;  
  22.     int i,j;  
  23. //定义base64编码表  
  24.     unsigned charchar *base64_table="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";  
  25.   
  26. //计算经过base64编码后的字符串长度  
  27.     str_len=strlen(str);  
  28.     if(str_len % 3 == 0)  
  29.         len=str_len/3*4;  
  30.     else  
  31.         len=(str_len/3+1)*4;  
  32.   
  33.     res=malloc(sizeof(unsigned char)*len+1);  
  34.     res[len]='\0';  
  35.   
  36. //以3个8位字符为一组进行编码  
  37.     for(i=0,j=0;i<len-2;j+=3,i+=4)  
  38.     {  
  39.         res[i]=base64_table[str[j]>>2]; //取出第一个字符的前6位并找出对应的结果字符  
  40.         res[i+1]=base64_table[(str[j]&0x3)<<4 | (str[j+1]>>4)]; //将第一个字符的后位与第二个字符的前4位进行组合并找到对应的结果字符  
  41.         res[i+2]=base64_table[(str[j+1]&0xf)<<2 | (str[j+2]>>6)]; //将第二个字符的后4位与第三个字符的前2位组合并找出对应的结果字符  
  42.         res[i+3]=base64_table[str[j+2]&0x3f]; //取出第三个字符的后6位并找出结果字符  
  43.     }  
  44.   
  45.     switch(str_len % 3)  
  46.     {  
  47.         case 1:  
  48.             res[i-2]='=';  
  49.             res[i-1]='=';  
  50.             break;  
  51.         case 2:  
  52.             res[i-1]='=';  
  53.             break;  
  54.     }  
  55.   
  56.     return res;  
  57. }  
  58.   
  59. unsigned charchar *base64_decode(unsigned charchar *code)  
  60. {  
  61. //根据base64表,以字符找到对应的十进制数据  
  62.     int table[]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,62,0,0,0,63,52,53,54,55,56,57,58,59,60,61,0,0,0,0,0,0,0,0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,0,0,0,0,0,0,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,51};  
  63.     long len;  
  64.     long str_len;  
  65.     unsigned charchar *res;  
  66.     int i,j;  
  67.   
  68. //计算解码后的字符串长度  
  69.     len=strlen(code);  
  70. //判断编码后的字符串后是否有=  
  71.     if(strstr(code,"=="))  
  72.         str_len=len/4*3-2;  
  73.     else if(strstr(code,"="))  
  74.         str_len=len/4*3-1;  
  75.     else  
  76.         str_len=len/4;  
  77.   
  78.     res=malloc(sizeof(unsigned char)*str_len+1);  
  79.     res[str_len]='\0';  
  80.   
  81. //以4个字符为一位进行解码  
  82.     for(i=0,j=0;i < len-2;j+=3,i+=4)  
  83.     {  
  84.         res[j]=((unsigned char)table[code[i]])<<2 | (((unsigned char)table[code[i+1]])>>4); //取出第一个字符对应base64表的十进制数的前6位与第二个字符对应base64表的十进制数的后2位进行组合  
  85.         res[j+1]=(((unsigned char)table[code[i+1]])<<4) | (((unsigned char)table[code[i+2]])>>2); //取出第二个字符对应base64表的十进制数的后4位与第三个字符对应bas464表的十进制数的后4位进行组合  
  86.         res[j+2]=(((unsigned char)table[code[i+2]])<<6) | ((unsigned char)table[code[i+3]]); //取出第三个字符对应base64表的十进制数的后2位与第4个字符进行组合  
  87.     }  
  88.   
  89.     return res;  
  90.   
  91. }  
  92.   
  93. /*一个测试程序*/  
  94. #include "base64.h"  
  95. #include <stdio.h>  
  96.   
  97. int main(int argc,charchar **argv)  
  98. {  
  99.     if(strcmp(argv[1],"-d") == 0)  
  100.         printf("%s\n",base64_decode(argv[2]));  
  101.     else  
  102.         printf("%s\n",base64_encode(argv[1]));  
  103.   
  104.     return 0;  
  105. }  
  106.   
  107. 编译  
  108. gcc -o test test.c base64.c  
  109.   
  110. ./test Brisk  
  111. QnJpc2s=  
  112. ./test -d QnJpc2s=  
  113. Brisk  
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值