`
piperzero
  • 浏览: 3475699 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
文章分类
社区版块
存档分类
最新评论

Base64编码介绍及基于Java语言实现

 
阅读更多

Base64编码介绍及基于Java语言实现

Base64编码被设计用来表示任意顺序的八位一个字节单元,允许使用大写和小写字符而不要求内容可读。65US-ASCII子集被用来表示Base64编码,所以每六位表示一个可以打印的Base64的字符,第65字符’=’被用来表示特殊处理的功能字符,后面会详细说明。

预编码处理一次输入24位(三个字符)作为一组,输出的是四个Base64的编码字符24位(每个字符六位)。编码处理遵循从左到右的顺序,一个24位的输入可以看作是三个八位的字符组,

编码以后,24位的字符则被转换为四个6位的字符组,每个字符被翻译成Base64字符表中的匹配的一个字符。

每六位都被转换成为一个index数值去匹配Base64编码表中的一个字符输出,Base64的字符表如下:

Value Encoding Value Encoding Value Encoding Value Encoding

0 A 17 R34 i 51 z

1 B 18 S 35 j 52 0

2 C 19 T36 k 53 1

3 D 20 U 37 l 54 2

4 E 21 V 38 m 55 3

5 F 22 W 39 n 56 4

6 G 23 X 40 o 57 5

7 H 24 Y 41 p 58 6

8 I 25 Z 42 q 59 7

9 J 26 a 43 r 60 8

10 K 27 b 44 s 61 9

11 L 28 c 45 t 62 +

12 M 29 d 46 u 63 /

13 N 30 e 47 v

14 O 31 f 48 w (pad) =

15 P32 g 49 x

16 Q 33 h 50 y

字符’=’作为填充字符,对于任何不满足24为输入的编码字符,以零做作为填充成为24位(三字节)输入,对于任何输出的编码字符小于24位的都应当以’=’字符填充。对于以下情形:

Ø最后输入正好是24位的编码字符,则不需要’=’字符填充,输出即为四个Base64字符编码.

Ø最后输入只有八位的编码字符,则需要填充两个特殊字符’=’,输出即为两个Base64字符编码加上两个’=’.

Ø最后输入只有十六位的编码字符,则需要填充一个特殊字符’=’,输出即为三个Base64字符加上一个’=’.

把一个二进制的数据转换为Base64的编码,下面的例子就说明Base64怎么编码三个字节的二进制数据的。

+--first octet--+-second octet--+--third octet--+

|7 6 5 4 3 2 1 0|7 6 5 4 3 2 1 0|7 6 5 4 3 2 1 0|

+-----------+---+-------+-------+---+-----------+

|5 4 3 2 1 0|5 4 3 2 1 0|5 4 3 2 1 0|5 4 3 2 1 0|

+--1.index--+--2.index--+--3.index--+--4.index--+

许多网络程序中用来传输二进制文件也经常采用Base64的编码算法,一个最典型的例子就是Email的应用协议SMPT发送MIME数据的时候就常采用Base64来发送,而且已经被写入标准的RFC中,在Java语言中,JDK已经提供了Base64API

下面基于Java实现的Base64的编码和解码程序,相比于JDK6中把Base64编码和解码分在两个不同的API Class中,因此还把许多人搞得很晕来说,也许本人的实现会略微方便点。

package com.gloomyfish.smtp.util;


public class Base64Coder {
	
	public final static char[] base64_alphabet = new char[]{
            'A','B','C','D','E','F','G','H','I','J','K','L','M','N','O','P','Q','R','S'
            ,'T','U','V','W','X','Y','Z','a','b','c','d','e','f','g','h','i','j','k','l'
            ,'m','n','o','p','q','r','s','t','u','v','w','x','y','z','0','1','2','3','4'
            ,'5','6','7','8','9','+','/','='
	};
	
	public static String encode(String content) {
		
		byte[] data = content.getBytes();
		int length = data.length;
		byte[] char_array_3 = new byte[]{0, 0, 0};
		byte[] char_array_4 = new byte[]{'=','=','=','='};
		String retContent = "";
		int i = 0;
		int j = 0;
		int reversePos = 0;
		while(length > 0) {
			length--;
			char_array_3[i++] = data[reversePos++];
			if(i==3) {
				char_array_4[0] = (byte)((char_array_3[0] & 0xfc) >> 2); // convert the char
			    char_array_4[1] = (byte)(((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4));
			    char_array_4[2] = (byte)(((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6));
			    char_array_4[3] = (byte)(char_array_3[2] & 0x3f);
			    for(i = 0; (i <4) ; i++)
			    	retContent += base64_alphabet[char_array_4[i]];
			    i = 0;
			}
		}
		
		// handling the last input content
		  if (i > 0 )
		  {
		    for(j = i; j < 3; j++)
		      char_array_3[j] = 0; // padding of zero

		    char_array_4[0] = (byte)((char_array_3[0] & 0xfc) >> 2); // right shift
		    char_array_4[1] = (byte)(((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4));
		    char_array_4[2] = (byte)(((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6));
		    char_array_4[3] = (byte)(char_array_3[2] & 0x3f);

		    for (j = 0; (j < i + 1); j++)
		    	retContent += base64_alphabet[char_array_4[j]];

		    while((i++ < 3)) // padding of '=' of output string
		    	retContent += '=';

		  }
		return retContent;
	}
	
	public static String decode(String enContent) {
		byte[] data = enContent.getBytes();
		int i = 0, j = 0, enCode = 0;
		int mLength = data.length;
		byte[] char_array_4 = new byte[4];
		byte[] char_array_3 = new byte[3];
		String retContent = "";

		// filter out the padding '=' chars
		  while (mLength > 0 && (((char)data[enCode]) != '=') && isBase64((char)data[enCode])) 
			{
			  mLength--;
			  char_array_4[i++] = data[enCode++];
			  if (i ==4) {
		      for (i = 0; i <4; i++)
		        char_array_4[i] = findChar((char)char_array_4[i]);

		      char_array_3[0] = (byte)((char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4));
		      char_array_3[1] = (byte)(((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2));
		      char_array_3[2] = (byte)(((char_array_4[2] & 0x3) << 6) + char_array_4[3]);

		      for (i = 0; (i < 3); i++)
		    	  retContent += (char)char_array_3[i];
		      i = 0;
		    }
		  }

		  // last content handling
		  if (i > 0) 
			{
		    for (j = i; j <4; j++)
		      char_array_4[j] = 0;

		    for (j = 0; j <4; j++)
		      char_array_4[j] = findChar((char)char_array_4[j]);

		    char_array_3[0] = (byte)((char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4));
		    char_array_3[1] = (byte)(((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2));
		    char_array_3[2] = (byte)(((char_array_4[2] & 0x3) << 6) + char_array_4[3]);

		    for (j = 0; (j < i - 1); j++) 
		    	retContent += (char)char_array_3[j];
		  }

		  return retContent;
	}
	
	public static boolean isBase64(char c) 
	{
		boolean base64 = false;
		for(int i=0; i<64; i++) {
			if( c == base64_alphabet[i]) {
				base64 = true;
				break;
			}
		}
	  return base64;
	}
	
	public static byte findChar(char x) {
		byte index = 64; // 65th char '='
		for(int i=0; i<64; i++) {
			if( x == base64_alphabet[i]) {
				index = (byte)i;
				break;
			}
		}
		return index;
	}
	
	/**
	 * <p> test data and result should like below output , RFC4648 Sample </p>
	 * 	BASE64("") = ""
	 *	BASE64("f") = "Zg=="
	 *	BASE64("fo") = "Zm8="
	 *	BASE64("foo") = "Zm9v"
	 *	BASE64("foob") = "Zm9vYg=="
	 *	BASE64("fooba") = "Zm9vYmE="
	 *	BASE64("foobar") = "Zm9vYmFy"
	 *
	 *
	 * @param args
	 */
	public static void main(String[] args) {
		// BASE64Encoder coder = new BASE64Encoder();
		// System.out.println(coder.encode("foobar".getBytes()));
		
		System.out.println("#--------------encode---------------#");
		System.out.println(encode(""));
		System.out.println(encode("f"));
		System.out.println(encode("fo"));
		System.out.println(encode("foo"));
		System.out.println(encode("foob"));
		System.out.println(encode("fooba"));
		System.out.println(encode("foobar"));
		System.out.println(encode("123456789sS{1}quot;));
		System.out.println("#--------------decode---------------#");
		System.out.println(decode(""));
		System.out.println(decode("Zg=="));
		System.out.println(decode("Zm8="));
		System.out.println(decode("Zm9v"));
		System.out.println(decode("Zm9vYg=="));
		System.out.println(decode("Zm9vYmE="));
		System.out.println(decode("Zm9vYmFy"));
		System.out.println(decode("MTIzNDU2Nzg5c1Mk"));
		
	}

}


分享到:
评论

相关推荐

    基于SMTP邮件截获与还原技术研究.pdf

    1.4.2 Base64编码/解码的算法实现 10 1.4.3 MIME简介 11 1.4.4 MIME邮件头的组成 12 1.4.5 MIME的编码格式 12 1.5 电子邮件的编写与分析 14 1.5.1 Email的封装过程 14 1.5.2 Email的传输过程 16 1.5.3 信封头 18 第...

    java课程设计 自己精心做的

    通过本实验,能让我们更好地理解并掌握网络编程技术这门课的相关知识:理解电子邮件的原理,了解TCP/IP协议、SMTP协议(即简易邮件传输协议),了解BASE64编码算法、网络数据截获工具JPCAP,以及用JAVA语言实现基于SMTP...

    java面试题

    84.8. 将一个键盘输入的数字转化成中文输出(例如:输入1234567,输出:一百二拾三万四千五百六拾七),请用java语言编一段程序实现! 114 84.9. 题目1:用1、2、2、3、4、5这六个数字,用java写一个main函数,打印出所有...

    【白雪红叶】JAVA学习技术栈梳理思维导图.xmind

    java语言 语言语法基础 异常 泛型 内部类 反射 序列化 nIo 匿名类 包装类 优先级 引用 语言工具类库 容器类 集合 链表 map 工具类 系统类 日期类 数字类 字符串+正则 流 字符流 字节流 ...

    sagacity-sqltoy:基于Java语言比mybatis更实用的orm框架,支持mysql,oracle,postgresql,sqlserver,db2,dm,mongodb,elasticsearch,tidb,guassdb,kingbase,oceanbase,greenplum

    docs /睿智平台SqlToy4.17使用手册.doc范例演示项目快速上手主要功能项目阅读其readme.md学习POJO和DTO严格分层演示项目分片分库分表演示nosql演示(mongo和elasticsearch) sqltoy基于xml配置演示编码云地址:...

    java 面试题 总结

    它是基于Java的远程方法调用(RMI)技术的,所以EJB可以被远程访问(跨进程、跨计算机)。但EJB必须被布署在诸如Webspere、WebLogic这样的容器中,EJB客户从不直接访问真正的EJB组件,而是通过其容器访问。EJB容器是...

    java版分销系统源码-NettyRPC:NettyRPC是基于Netty的高性能javarpc服务器,使用kryo,hessian,prot

    NettyRPC基于Java语言进行编写,网络通讯依赖Netty。 RPC服务端采用线程池对RPC调用进行异步回调处理。 服务定义、实现,通过Spring容器进行加载、卸载。 消息网络传输除了JDK原生的对象序列化方式,还支持目前主流...

    MaxKB 是一款基于 LLM 大语言模型的知识库问答系统

    MaxKB 是一款基于 LLM 大语言模型的知识库问答系统。MaxKB = Max Knowledge Base,旨在成为企业的最强大脑。支持直接上传文档、自动爬取在线文档,支持文本自动拆分、向量化,智能问答交互体验好;支持零编码快速...

    Jakarta commons docs API CHM 格式

    包括一些语音编码器, Hex, Base64, 以及URL encoder. commons-collections 扩充了 java.util.collection 接口 commons-compress 操作压缩文件的 API commons-configuration 可以从xml、properties、JNDI、JDBC、...

    比较全面的:Jakarta-commons jar包(附: chm参考手册 & 资源简介)

    包括一些语音编码器, Hex, Base64, 以及URL encoder. commons-collections 扩充了 java.util.collection 接口 commons-compress 操作压缩文件的 API commons-configuration 可以从xml、properties、JNDI、JDBC、...

    Java开发实战1200例(第1卷).(清华出版.李钟尉.陈丹丹).part3

    实例241 基于Base64编码 304 实例242 基于Base64解码 305 实例243 发送简单的Email 306 实例244 发送带附件的Email 308 实例245 读取XML文件属性 310 第3篇 窗体与控件应用 第11章 窗体设计 314 11.1 设置窗体位置 ...

    超级有影响力霸气的Java面试题大全文档

    它是基于Java的远程方法调用(RMI)技术的,所以EJB可以被远程访问(跨进程、跨计算机)。但EJB必须被布署在诸如Webspere、WebLogic这样的容器中,EJB客户从不直接访问真正的EJB组件,而是通过其容器访问。EJB容器是...

    apache-commons源码及jar文件

    包括一些语音编码器, Hex, Base64, 以及URL encoder. Collections Commons-Collections 提供一个类包来扩展和增加标准的 Java Collection框架 Configuration Commons-Configuration 工具对各种各式的配置和参考...

    branscha:配置文件存储库

    储存库 应用领域 ... 将字符串编码解码为base64格式或从base64格式解码。 Perl 使用自定义命令控制台应用程序。 项目模板 基本的 :开始。 :Java启动器。 :Java命令。 :Common Lisp启动器。

    java经典面试2010集锦100题(不看你后悔)

    D) Java语言中的多态的含义可以表达为:对外一个接口,内部多种实现。Java语言支持两种多态:运行时多态和编译时多态。 题目8:a 程序如下: public class Demo extends Base { private int count; public Demo()...

    appframework:从 code.google.compappframework 自动导出

    概述 appframework 是一个基于的 java web 应用程序 appframework 适用于 Java ...提供基本的编码和加密方法,例如 md5、base64、dsa、rsa、aes、des、xxtea 为 log4j 和 common-logging 提供包装日志 提供常用工具

    asp.net知识库

    常用编码工具类,支持base64,md5,des,crc32 也谈谈技术面试 在C#里把ArrayList转换为Array 或 把Array转换为ArrayList C# 2.0 在.NET 2.0中,让你的组件也可以绑定 .NET20 一种简单的窗口控件UI状态控制方法 翻译MSDN...

Global site tag (gtag.js) - Google Analytics