/* * Copyright (C) 2012 www.amsoft.cn * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.ab.util; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.OutputStream; import java.io.UnsupportedEncodingException; // TODO: Auto-generated Javadoc /** * © 2012 amsoft.cn * 名称:AbBase64.java * 描述:Base64工具类. * * @author 还如一梦中 * @version v1.0 * @date:2011-11-10 下午11:52:13 */ public class AbBase64 { private static final char[] legalChars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/" .toCharArray(); /** The Constant base64EncodeChars. */ private static final char[] base64EncodeChars = 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', '+', '/' }; /** The Constant base64DecodeChars. */ private static final byte[] base64DecodeChars = new byte[] { -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 62, -1, -1, -1, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, -1, -1, -1, -1, -1, -1, -1, 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, -1, -1, -1, -1, -1, -1, 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, -1, -1, -1, -1, -1 }; /** * Base64编码 * * @param data * @return */ public static String encode(byte[] data) { int start = 0; int len = data.length; StringBuffer buf = new StringBuffer(data.length * 3 / 2); int end = len - 3; int i = start; while (i <= end) { int d = ((((int) data[i]) & 0x0ff) << 16) | ((((int) data[i + 1]) & 0x0ff) << 8) | (((int) data[i + 2]) & 0x0ff); buf.append(legalChars[(d >> 18) & 63]); buf.append(legalChars[(d >> 12) & 63]); buf.append(legalChars[(d >> 6) & 63]); buf.append(legalChars[d & 63]); i += 3; } if (i == start + len - 2) { int d = ((((int) data[i]) & 0x0ff) << 16) | ((((int) data[i + 1]) & 255) << 8); buf.append(legalChars[(d >> 18) & 63]); buf.append(legalChars[(d >> 12) & 63]); buf.append(legalChars[(d >> 6) & 63]); buf.append("="); } else if (i == start + len - 1) { int d = (((int) data[i]) & 0x0ff) << 16; buf.append(legalChars[(d >> 18) & 63]); buf.append(legalChars[(d >> 12) & 63]); buf.append("=="); } return buf.toString(); } /** * Base64解码 * * @param s * @return */ public static byte[] decode(String s) { ByteArrayOutputStream bos = new ByteArrayOutputStream(); try { decode(s, bos); } catch (IOException e) { throw new RuntimeException(); } byte[] decodedBytes = bos.toByteArray(); try { bos.close(); bos = null; } catch (IOException ex) { System.err.println("Error while decoding BASE64: " + ex.toString()); } return decodedBytes; } private static void decode(String s, OutputStream os) throws IOException { int i = 0; int len = s.length(); while (true) { while (i < len && s.charAt(i) <= ' ') i++; if (i == len) break; int tri = (decode(s.charAt(i)) << 18) + (decode(s.charAt(i + 1)) << 12) + (decode(s.charAt(i + 2)) << 6) + (decode(s.charAt(i + 3))); os.write((tri >> 16) & 255); if (s.charAt(i + 2) == '=') break; os.write((tri >> 8) & 255); if (s.charAt(i + 3) == '=') break; os.write(tri & 255); i += 4; } } private static int decode(char c) { if (c >= 'A' && c <= 'Z') return ((int) c) - 65; else if (c >= 'a' && c <= 'z') return ((int) c) - 97 + 26; else if (c >= '0' && c <= '9') return ((int) c) - 48 + 26 + 26; else switch (c) { case '+': return 62; case '/': return 63; case '=': return 0; default: throw new RuntimeException("unexpected code: " + c); } } /** * Encode. * * @param str * the str * @param charsetName * the charset name * @return the string */ public static final String encode(String str, String charsetName) { return encode(str, charsetName, 0); } /** * Encode. * * @param str * the str * @param charsetName * the charset name * @param width * the width * @return the string */ public static final String encode(String str, String charsetName, int width) { byte[] data = null; try { data = str.getBytes(charsetName); } catch (UnsupportedEncodingException e) { e.printStackTrace(); return null; } int length = data.length; int size = (int) Math.ceil(length * 1.36); int splitsize = width > 0 ? size / width : 0; StringBuffer sb = new StringBuffer(size + splitsize); int r = length % 3; int len = length - r; int i = 0; int c; while (i < len) { c = (0x000000ff & data[i++]) << 16 | (0x000000ff & data[i++]) << 8 | (0x000000ff & data[i++]); sb.append(base64EncodeChars[c >> 18]); sb.append(base64EncodeChars[c >> 12 & 0x3f]); sb.append(base64EncodeChars[c >> 6 & 0x3f]); sb.append(base64EncodeChars[c & 0x3f]); } if (r == 1) { c = 0x000000ff & data[i++]; sb.append(base64EncodeChars[c >> 2]); sb.append(base64EncodeChars[(c & 0x03) << 4]); sb.append("=="); } else if (r == 2) { c = (0x000000ff & data[i++]) << 8 | (0x000000ff & data[i++]); sb.append(base64EncodeChars[c >> 10]); sb.append(base64EncodeChars[c >> 4 & 0x3f]); sb.append(base64EncodeChars[(c & 0x0f) << 2]); sb.append("="); } if (splitsize > 0) { char split = '\n'; for (i = width; i < sb.length(); i++) { sb.insert(i, split); i += width; } } return sb.toString(); } /** * Decode. * * @param str * the str * @param charsetName * the charset name * @return the string */ public static final String decode(String str, String charsetName) { byte[] data = null; try { data = str.getBytes(charsetName); } catch (UnsupportedEncodingException e) { e.printStackTrace(); return null; } int len = data.length; ByteArrayOutputStream buf = new ByteArrayOutputStream( (int) (len * 0.67)); int i = 0; int b1, b2, b3, b4; while (i < len) { do { if (i >= len) { b1 = -1; break; } b1 = base64DecodeChars[data[i++]]; } while (i < len && b1 == -1); if (b1 == -1) { break; } do { if (i >= len) { b2 = -1; break; } b2 = base64DecodeChars[data[i++]]; } while (i < len && b2 == -1); if (b2 == -1) { break; } buf.write((int) ((b1 << 2) | ((b2 & 0x30) >>> 4))); do { if (i >= len) { b3 = -1; break; } b3 = data[i++]; if (b3 == 61) { b3 = -1; break; } b3 = base64DecodeChars[b3]; } while (i < len && b3 == -1); if (b3 == -1) { break; } buf.write((int) (((b2 & 0x0f) << 4) | ((b3 & 0x3c) >>> 2))); do { if (i >= len) { b4 = -1; break; } b4 = data[i++]; if (b4 == 61) { b4 = -1; break; } b4 = base64DecodeChars[b4]; } while (b4 == -1); if (b4 == -1) { break; } buf.write((int) (((b3 & 0x03) << 6) | b4)); } try { return buf.toString(charsetName); } catch (UnsupportedEncodingException e) { e.printStackTrace(); return null; } } }