本文主要是介绍银行电子回单打印中Base64使用。,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
最近做银行电子回单打印时存在以下Base64的使用场景:
1:银行回单打印完毕后,生成一个bmp格式的图片,然后将该回单图片作为附件上传到国网的UDS非结构化中心,并挂接到银行交易流水记录对应的凭证中去。
2:回单打印时,由于交易流水记录的摘要字段中存在%等字符导致打印异常。
对于以上两种场景,均使用到了Base64编解码,首先对于第一种场景,回单打印完毕后,直接在客户端生成一个base64编码的字符串(即回单图片)。在服务端进行解码上传。同样第二种场景在打印前构造回单的数据(构造为一个xml)信息时对交易流水记录的摘要进行Base64编码,打印前对摘要进行解码。
/*** */
package utils;import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.UUID;import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.geronimo.mail.util.Base64;/*** Base64工具.* @author libaobao*/
public final class EncryBase64Util {private static Log LOG = LogFactory.getLog(EncryBase64Util.class);/*** 将base64字符串转换为字节流并存在服务端的临时目录.* @param base64String* @return 字节数组*/public static byte[] decodeToBytesAndSave(String base64String) {byte[] bytes = null;// 获取服务端的临时目录.final String tmpDir = getTmpDir();final String realDir = tmpDir + File.separator + "paysettle";// 创建临时目录.final File aCreateDir = new File(realDir);if (!aCreateDir.exists()) {aCreateDir.mkdirs();if (aCreateDir.exists()) {LOG.info("创建临时目录成功:" + aCreateDir.getAbsolutePath());} else {LOG.info("创建临时目录失败!");throw new RuntimeException("创建临时目录失败!");}}// 保存到临时文件下.final String tmpFileName = newUUID() + ".jpg";final String tmpFileFullName = realDir + File.separator + tmpFileName;if (LOG.isInfoEnabled()) {LOG.info("文件名:" + tmpFileFullName);}// 确定写出文件的位置final File file = new File(tmpFileFullName);//建立输出字节流FileOutputStream fos = null;try {fos = new FileOutputStream(file);bytes = decodeToBytes(base64String);fos.write(bytes);LOG.info("写入成功!");} catch (FileNotFoundException e) {LOG.error("写入文件失败", e);} catch (IOException e) {LOG.error("写入文件失败", e);} finally {if (fos != null) {// 为了节省IO流的开销,需要关闭try {fos.close();} catch (IOException e) {LOG.error("关闭失败", e);}}}return bytes;} /*** 获取服务端可以写入的一个临时目录,这里写死C盘* @return*/private static String getTmpDir() { return "C://";}
/*** 生成uuid,可用于生成数据库的唯一主键. * @return uuid*/private static String newUUID() {UUID uuid = UUID.randomUUID();final String s = uuid.toString();return s.substring(0,8)+s.substring(9,13)+s.substring(14,18)+s.substring(19,23)+s.substring(24); }/*** 将base64字符串转换为字节流并存在服务端的临时目录.* @param base64String* @return 字节数组*/private static byte[] decodeToBytes(String base64String) {// TODO Auto-generated method stubreturn Base64.decode(base64String);}/*** @param args*/public static void main(String[] args) {//LOG.error(newUUID()); String content = "iVBORw0KGgoAAAANSUhEUgAAA...此处省略1万字";decodeToBytesAndSave(content);}}
上述是第一中场景的模拟使用。对于第二种场景直接在客户端处理,客户端需自行实现base64的编解码,服务端无需自己实现,jdk,Spring均有实现。客户端实现见:http://www.cnblogs.com/ryans/p/6512631.html
/*** 对摘要和用途用base64加密,避免摘要中的特殊字符导致打印异常.*/
function buildBase64(summary) {return "[base64]" + desUitls.encodeBase64(summary);}
Base64编码由来
http://www.cnblogs.com/chengxiaohui/articles/3951129.html
为什么会有Base64 编码呢?因为有些网络传送渠道并不支持所有的字节,例如传统的邮件只支持可见字符的传送,像ASCII码的控制字符就 不能通过邮件传送。这样用途就受到了很大的限制,比如图片二进制流的每个字节不可能全部是可见字符,所以就传送不了。最好的方法就是在不改变传统协议的情 况下,做一种扩展方案来支持二进制文件的传送。把不可打印的字符也能用可打印字符来表示,问题就解决了。Base64编码应运而生,Base64就是一种 基于64个可打印字符来表示二进制数据的表示方法。
Base64的索引表,Base编码的字符选用了”A-Z、a-z、0-9、+、/” 64个可打印字符。数值代表字符的索引,这个是标准Base64协议规定的,不能更改。64个字符用6个bit位就可以全部表示,一个字节有8个bit 位,剩下两个bit就浪费掉了,这样就不得不牺牲一部分空间了。这里需要弄明白的就是一个Base64字符是8个bit,但是有效部分只有右边的6个 bit,左边两个永远是0。
但是转换到最后你发现不够三个字节了怎么办呢?愿望终于实现了,我们可以用两 个Base64来表示一个字符或用三个Base64表示两个字符,像下图的A对应的第二个Base64的二进制位只有两个,把后边的四个补0就是了。所以 A对应的Base64字符就是QQ。上边已经说过了,原则是Base64字符的最小单位是四个字符一组,那这才两个字 符,后边补两个”=”吧。其实不用”=”也不耽误解码,之所以用”=”,可能是考虑到多段编码后的Base64字符串拼起来也不会引起混淆。由此可见 Base64字符串只可能最后出现一个或两个”=”,中间是不可能出现”=”的。下图中字符”BC”的编码过程也是一样的。
这篇关于银行电子回单打印中Base64使用。的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!