Base64编码算法

Base64是MIME邮件中常用的编码方式之一。它的主要思想是将输入的字符串或数据编码成只含有{‘A‘-‘Z‘, ‘a‘-‘z‘, ‘0‘-‘9‘, ‘+‘, ‘/‘}这64个可打印字符的串,故称为“Base64”。 Base64编码的方法是,将输入数据流每次取6 bit,用此6 bit的值(0-63)作为索引去查表,输出相应字符。这样,每3个字节将编码为4个字符(3×8 → 4×6);不满4个字符的以‘=‘填充。

编码过程是这样的,第一个字符通过右移2位获得第一个目标字符的base64表位置,根据这个数值取到表上相应的字符,就是第一个目标字符,然后将第一个字符左移6位加上第二个字符右移4位,即获得第二个目标字符,再将第二个字符左移4位加上第三个字符右移6位,获得第三个目标字符,最后取第三个字符的右6位即获得第四个目标字符.

编码解码函数:

#i nclude <stdio.h>
#i nclude <stdlib.h>
#i nclude <string.h>

unsigned char * base64="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";



unsigned char * encode(unsigned char *src,int srclen)
{
int n,buflen,i,j;
unsigned char *buf;
static unsigned char *dst;

buf=src;
buflen=n=srclen;
dst=(unsigned char*)malloc(buflen/3*4+3);
memset(dst,0,buflen/3*4+3);
for(i=0,j=0;i<buflen;i+=3,j+=4) {
dst[j]=(buf[i]&0xFC)>>2;
dst[j+1]=((buf[i]&0x03)<<4) + ((buf[i+1]&0xF0)>>4);
dst[j+2]=((buf[i+1]&0x0F)<<2) + ((buf[i+2]&0xC0)>>6);
dst[j+3]=buf[i+2]&0x3F;
}
if(n%3==1) {
j-=4,i-=3;
dst[j]=(buf[i]&0xFC)>>2;
dst[j+1]=((buf[i]&0x03)<<4);
dst[j+2]=64;
dst[j+3]=64;
j+=4;
}
else if(n%3==2) {
j-=4,i-=3;
dst[j]=(buf[i]&0xFC)>>2;
dst[j+1]=((buf[i]&0x03)<<4) + ((buf[i+1]&0xF0)>>4);
dst[j+2]=((buf[i+1]&0x0F)<<2);
dst[j+3]=64;
j+=4;
}
for(i=0;i<j;i++) /* map 6 bit value to base64 ASCII character */
dst[i]=base64[dst[i]];
dst[j]=0;
return dst;
}

unsigned char *decode(unsigned char *src,int * len)
{
int n,i,j,pad;
unsigned char *p;
static unsigned char *dst;
*len=0;
pad=0;
n=strlen(src);
while(n>0&&src[n-1]=='=') {
src[n-1]=0;
pad++;
n--;
}
for(i=0;i<n;i++) { /* map base64 ASCII character to 6 bit value */
p=strchr(base64,src[i]);
if(!p)
break;
src[i]=p-base64;
}

dst=malloc(n*3/4+1);
memset(dst,0,n*3/4+1);
for(i=0,j=0;i<n;i+=4,j+=3) {
dst[j]=(src[i]<<2) + ((src[i+1]&0x30)>>4);
dst[j+1]=((src[i+1]&0x0F)<<4) + ((src[i+2]&0x3C)>>2);
dst[j+2]=((src[i+2]&0x03)<<6) + src[i+3];
*len+=3;
}
*len-=pad;
return dst;
}



#define BUF_SIZE 57
#define LINE_SIZE 80



int main(int argc,char * argv[])
{
FILE * fp,*fpo;
unsigned char * cdc;
unsigned char buf[BUF_SIZE];
unsigned char line[LINE_SIZE];
char * nl,*eol;
int r;
if(argc<3) {
fprintf(stderr,"Uasge: base64 [de] filename/n");
return -1;
}



if(argv[1][0]=='e') { /*encode*/
fp=fopen(argv[2],"rb");
if(fp==NULL) {
fprintf(stderr,"Can not open file %s/n",argv[2]);
return -1;
}
while((r=fread(buf,sizeof(unsigned char),BUF_SIZE,fp))>0) {
cdc=encode(buf,r);
fprintf(stdout,"%s/n",cdc);
}
fclose(fp);
}
else if(argv[1][0]=='d') { /*decode*/
if(argc<4) {
fprintf(stderr,"Uasge: base64 d filename outname/n");
return -1;
}
fp=fopen(argv[2],"r");
fpo=fopen(argv[3],"wb");
if(fp==NULL) {
fprintf(stderr,"Can not open file %s/n",argv[2]);
return -1;
}
while(fgets(line,LINE_SIZE,fp)) {
nl=strchr(line,'/n');
eol=strchr(line,'/r');
if(nl)
*nl=0;
if(eol)
*eol=0;
r=strlen(line);
if(r!=0) {
cdc=decode(line,&r);
fwrite(cdc,1,r,fpo);
}
}
fclose(fp);
}
else {
fprintf(stderr,"Uasge: base64 [de] filename/n");
return -1;
}

return 0;
}
 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值