Base64編碼


BASE64 編碼是一種常用的字符編碼,在很多地方都會用到。但base64 不是安全領域下的加密解密算法。能起到安全作用的效果很差,而且很容易破解, 他核心作用應該是【傳輸數據的正確性】,有些網關或系統只能使用ASCII字符。Base64就是 用來將非ASCII字符的數據轉換成ASCII字符的一種方法,而且base64 特別適合在http,mime協議下快速傳輸數據

Base64編解碼原理

Base64使用【字母az AZ數字 09和+/】這64個字符
編碼原理是將3個字節轉換成4個字節
(3 X 8) = 24 = (4 X 6) 
先讀入3個字節每讀一個字節左移8位再右移四次每次6位這樣就有4個字節了。
這樣還不能保證得到的字符都是可見字符為了達到此目的Base64制定了一個編碼表進行統一的轉換。
碼表的大小為2^6=64這也是Base64名稱的由來。
當剩下的字符數量不足3個字節時則應使用0進行填充相應的輸出字符則使用'='占位因此編碼后輸出的文本末尾可能會出現1至2個'='。

解碼原理是將4個字節轉換成3個字節
先讀入4個6位(用或運算)每次左移6位再右移3次每次8位這樣就還原了。

Base64編碼表

    0 A         9   J       17 R           26 a       34 i        43 r          52 0 60 8        
   1 B         10 K       18 S          27 b       35 j        44 s         53 1 61 9       
   2 C         11 L       19 T          28 c        36 k       45 t         54 2 62 +       
   3 D        12 M      20 U          29 d       37 l        46 u        55 3 63 /       
   4 E         13 N      21 V          30 e        38 m      47 v       56 4              
   5 F         14 O      22 W         31 f         39 n       48 w       57 5
   6 G        15 P       23 X          32 g        40 o       49 x        58 6
   7 H        16 Q      24 Y          33 h        41 p       50 y        59 7
   8 I        25 Z        42 q       51 z    

JDK里面實現Base64的API

  • 在JDK1.6之前,JDK核心類一直沒有Base64的實現類,有人建議用JDK里面的【sun.misc.BASE64Encoder】和 【sun.misc.BASE64Decoder】,使用它們的優點就是不需要依賴第三方類庫,缺點就是可能在未來版本會被刪除(用maven編譯會發出警告),而且性能不佳。
  • JDK1.6中添加了另一個Base64的實現【javax.xml.bind.DatatypeConverter】兩個靜態方法【parseBase64Binary】和【printBase64Binary】,隱藏在javax.xml.bind包下面,不被很多開發者知道。
  • 在Java8在java.util包下面實現了BASE64編解碼API,而且性能不俗,API也簡單易懂,下面展示下這個類的使用例子。

java.util.Base64

該類提供了一套靜態方法獲取下面三種BASE64編解碼器:
  • Basic編碼:是標准的BASE64編碼,用於處理常規的需求
  • URL編碼:使用下划線替換URL里面的反斜線“/”
  • MIME編碼:使用基本的字母數字產生BASE64輸出,而且對MIME格式友好:每一行輸出不超過76個字符,而且每行以“\r\n”符結束。

示例代碼

public class Base64Test {
    public static void main(String[] args) throws UnsupportedEncodingException {
        String str = "/白乾濤";
        String charset = "UTF-8";
        //Basic編碼:是標准的BASE64編碼,用於處理常規的需求
        String encStr = Base64.getEncoder().encodeToString(str.getBytes(charset));
        byte[] asBytes = Base64.getDecoder().decode(encStr);
        System.out.println("編碼【" + encStr + "】解碼【" + new String(asBytes, charset) + "】")//編碼【L+eZveS5vua2mw==】解碼【/白乾濤】
        //URL編碼:使用下划線替換URL里面的反斜線“/”
        String encStr2 = Base64.getUrlEncoder().encodeToString(str.getBytes(charset));
        byte[] asBytes2 = Base64.getUrlDecoder().decode(encStr2);
        System.out.println("編碼【" + encStr2 + "】解碼【" + new String(asBytes2, charset) + "】")//編碼【L-eZveS5vua2mw==】解碼【/白乾濤】
        //MIME編碼:使用基本的字母數字產生BASE64輸出
        String encStr3 = Base64.getMimeEncoder().encodeToString(str.getBytes(charset));
        byte[] asBytes3 = Base64.getMimeDecoder().decode(encStr3);
        System.out.println("編碼【" + encStr3 + "】解碼【" + new String(asBytes3, charset) + "】")//編碼【L+eZveS5vua2mw==】解碼【/白乾濤】
        //MIME編碼對MIME格式友好:每一行輸出不超過76個字符,而且每行以“\r\n”符結束。
        StringBuilder sb = new StringBuilder();
        for (int t = 0; t < 5; ++t) {
            sb.append(UUID.randomUUID().toString());
        }
        System.out.println("原始字符【" + sb.toString() + "】");
        byte[] toEncode = sb.toString().getBytes(charset);
        System.out.println(Base64.getMimeEncoder().encodeToString(toEncode));
    }
}

第三方實現Base64的API

  • 常用的是Apache Commons Codec library里面的【org.apache.commons.codec.binary.Base64】
  • 第二個便是Google Guava庫里面的【com.google.common.io.BaseEncoding.base64()】這個靜態方法
  • 第三個是【net.iharder.Base64】,這個jar包就一個類
  • 最后一個,號稱Base64編碼速度最快的MigBase64,而且是10年前的實現
org.apache.commons.codec.binary.Base64的結構

Base64編碼性能測試

上面講了一共7種實現Base64編碼,Jdk里面3種,第三方實現4種,一旦有選擇,則有必要將他們進行一次高低對比,
一切都很明顯了,從上面看出,sun的表現不是很好,IHarder和MigBase64性能可以接受,傳說MigBase64性能第一,那也是過去了,在這次測試結果中, 新的java8 base64運行速度最好,javaXml表現次之。

如果你需要一個性能好可靠的Base64編解碼器不要找JDK外面的了java8里面的java.util.Base64以及java6中隱藏很深的javax.xml.bind.DatatypeConverter都是不錯的選擇。





免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM