天天看點

Base64的java實作

Base64 使用US-ASCII子集的65個字元, 每個字元用6位表示 

是以"m"的Base64值為38, 二進制形式是 100110. 

對于文本串,編碼過程如下。例如"men": 

先轉成US-ASCII值. 

"m"十進制 109 

"e"十進制 101 

"n"十進制 110 

二進制 : 

m 01101101 

e 01100101 

n 01101110 

三個8位連起來是24位 

011011010110010101101110 

然後分成4個6位 

011011 010110 010101 101110 

現在得到4個值,十進制為 

27 22 21 46 

對應的 Base64 字元是 : 

b W V u 

編碼總是基于3個字元,進而産生4個Base64字元。 

如果隻剩1或2個字元,使用特殊字元"="補齊Base64的4字。 

如,編碼"me" 

01101101 01100101 

0110110101100101 

011011 010110 0101 

111111 (與,補足6位) 

011011 010110 010100 

b W U 

b W U = ("=" 補足4字元) 

于是 "bWU=" 就是"me"的Base64值. 

再如編碼 "m" 

01101101 

011011 01 

111111 

011011 010000 

b Q = = 

于是 "bQ==" 就是"m"的Base64值. 

值得注意的是,MIME規定一行最多76個字元.

import java.io.*; 

public class MIMEBase64 { 

static String BaseTable[] = { 

"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 void encode(String filename, BufferedWriter out) { 

try { 

File f = new File(filename); 

FileInputStream fin = new FileInputStream(filename); 

// 讀檔案到BYTE數組 

byte bytes[] = new byte[(int)(f.length())]; 

int n = fin.read(bytes); 

if (n < 1) return; // 沒有内容 

byte buf[] = new byte[4]; // base64 字元數組 

int n3byt = n / 3; // 3 bytes 組數 

int nrest = n % 3; // 分組後剩餘 bytes 

int k = n3byt * 3; // 

int linelength = 0; // 行長 

int i = 0; // 指針 

// 3-bytes 分組 ... 

while ( i < k ) { 

buf[0] = (byte)(( bytes[i] & 0xFC) >> 2); 

buf[1] = (byte)(((bytes[i] & 0x03) << 4) | 

((bytes[i+1] & 0xF0) >> 4)); 

buf[2] = (byte)(((bytes[i+1] & 0x0F) << 2) | 

((bytes[i+2] & 0xC0) >> 6)); 

buf[3] = (byte)( bytes[i+2] & 0x3F); 

send(out, BaseTable[buf[0]]); 

send(out, BaseTable[buf[1]]); 

send(out, BaseTable[buf[2]]); 

send(out, BaseTable[buf[3]]); 

if ((linelength += 4) >= 76) { 

send(out, "/r/n"); 

linelength = 0; 

i += 3; 

// 處理尾部 ... 

if (nrest==2) { 

// 2 bytes left 

buf[0] = (byte)(( bytes[k] & 0xFC) >> 2); 

buf[1] = (byte)(((bytes[k] & 0x03) << 4) | 

((bytes[k+1] & 0xF0) >> 4)); 

buf[2] = (byte)(( bytes[k+1] & 0x0F) << 2); 

else if (nrest==1) { 

// 1 byte left 

buf[0] = (byte)((bytes[k] & 0xFC) >> 2); 

buf[1] = (byte)((bytes[k] & 0x03) << 4); 

if (nrest > 0) { 

// 發送尾部 

if ((linelength += 4) >= 76) send(out, "/r/n"); 

send(out, BaseTable[buf[0]]); 

send(out, BaseTable[buf[1]]); 

// 

if (nrest==2) { 

send(out, BaseTable[buf[2]]); 

else { 

send(out, "="); 

send(out, "="); 

out.flush(); 

//這裡用到的send方法,請大家根據需要,自己寫。可以是把結果輸出到控制台,或發送郵件。 

catch (Exception e) { 

e.printStackTrace(); 

}