首页 > 其他分享 >[转]JDK1.6和jdk8实现BASE64编解码的API

[转]JDK1.6和jdk8实现BASE64编解码的API

时间:2023-12-15 21:33:22浏览次数:35  
标签:编解码 BASE64 String Base64 base64 API new byte final

原文出处: 成熟的毛毛虫的博客

BASE64 编码是一种常用的字符编码,在很多地方都会用到。但base64不是安全领域下的加密解密算法。能起到安全作用的效果很差,而且很容易破解,他核心作用应该是传输数据的正确性,有些网关或系统只能使用ASCII字符。Base64就是用来将非ASCII字符的数据转换成ASCII字符的一种方法,而且base64特别适合在http,mime协议下快速传输数据。

JDK里面实现Base64的API

在JDK1.6之前,JDK核心类一直没有Base64的实现类,有人建议用Sun/Oracle JDK里面的sun.misc.BASE64Encoder 和 sun.misc.BASE64Decoder,使用它们的优点就是不需要依赖第三方类库,缺点就是可能在未来版本会被删除(用maven编译会发出警告),而且性能不佳,后面会有性能测试。

JDK1.6中添加了另一个Base64的实现,javax.xml.bind.DatatypeConverter两个静态方法parseBase64Binary 和 printBase64Binary,隐藏在javax.xml.bind包下面,不被很多开发者知道。

在Java 8在java.util包下面实现了BASE64编解码API,而且性能不俗,API也简单易懂,下面展示下这个类的使用例子。

java.util.Base64

该类提供了一套静态方法获取下面三种BASE64编解码器:

1)Basic编码:是标准的BASE64编码,用于处理常规的需求

1 2 3 4 5 6 // 编码 String asB64 = Base64.getEncoder().encodeToString("some string".getBytes("utf-8")); System.out.println(asB64); // 输出为: c29tZSBzdHJpbmc= // 解码 byte[] asBytes = Base64.getDecoder().decode("c29tZSBzdHJpbmc="); System.out.println(new String(asBytes, "utf-8")); // 输出为: some string

2)URL编码:使用下划线替换URL里面的反斜线“/”

1 2 3 4 String urlEncoded = Base64.getUrlEncoder().encodeToString("subjects?abcd".getBytes("utf-8")); System.out.println("Using URL Alphabet: " + urlEncoded); // 输出为: Using URL Alphabet: c3ViamVjdHM_YWJjZA==

3)MIME编码:使用基本的字母数字产生BASE64输出,而且对MIME格式友好:每一行输出不超过76个字符,而且每行以“\r\n”符结束。

1 2 3 4 5 6 7 StringBuilder sb = new StringBuilder(); for (int t = 0; t < 10; ++t) {   sb.append(UUID.randomUUID().toString()); } byte[] toEncode = sb.toString().getBytes("utf-8"); String mimeEncoded = Base64.getMimeEncoder().encodeToString(toEncode); System.out.println(mimeEncoded);

第三方实现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年前的实现,到现在是否能保持这个称号,测一测便知道;

Base64编码性能测试

上面讲了一共7种实现Base64编码,Jdk里面3种,第三方实现4种,一旦有选择,则有必要将他们进行一次高低对比,性能测试是最直接的方式

首先来定义两个接口

1 2 3 4 5 6 7 8 9 10 private static interface Base64Codec     {         public String encode(final byte[] data);         public byte[] decode(final String base64) throws IOException;     }     private static interface Base64ByteCodec     {         public byte[] encodeBytes(final byte[] data);         public byte[] decodeBytes(final byte[] base64) throws IOException;     }

两个接口区别就是其中一个接口方法参数接收byte数组,返回byte数组,因为byte->byte相比String->byte或者byte->String性能上会快一点,所以区分两组来测试

1 2 3 4 private static final Base64Codec[] m_codecs = { new GuavaImpl(), new JavaXmlImpl(),         new Java8Impl(), new SunImpl(), new ApacheImpl(),new MiGBase64Impl(),new IHarderImpl() }; private static final Base64ByteCodec[] m_byteCodecs = {         new ApacheImpl(), new Java8Impl(),new MiGBase64Impl(),new IHarderImpl() };

从上面看出,其中支持byte->byte只有4中API;

7个Base64的实现类

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 26 27 28 29 private static class Java8Impl implements Base64Codec, Base64ByteCodec     {         private final Base64.Decoder m_decoder = Base64.getDecoder();         private final Base64.Encoder m_encoder = Base64.getEncoder();         @Override         public String encode(byte[] data) {             return m_encoder.encodeToString(data);         }         @Override         public byte[] decode(String base64) throws IOException {             return m_decoder.decode(base64);         }         public byte[] encodeBytes(byte[] data) {             return m_encoder.encode( data );         }         public byte[] decodeBytes(byte[] base64) throws IOException {             return m_decoder.decode( base64 );         }     }     private static class JavaXmlImpl implements Base64Codec //no byte[] implementation     {         public String encode(byte[] data) {             return DatatypeConverter.printBase64Binary( data );         }         public byte[] decode(String base64) throws IOException {             return DatatypeConverter.parseBase64Binary( base64 );         }     } ..............

后面代码基本就是各种API实现Base64的代码了,就不详细列出。

主要测试手段是,生成100M的随机数,分成100byte或者1000byte的块,然后将他们分别编码和解码,记录时间,如下方法

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 private static TestResult testByteCodec( final Base64ByteCodec codec, final List<byte[]> buffers ) throws IOException {         final List<byte[]> encoded = new ArrayList<byte[]>( buffers.size() );         final long start = System.currentTimeMillis();         for ( final byte[] buf : buffers )             encoded.add( codec.encodeBytes(buf) );         final long encodeTime = System.currentTimeMillis() - start;         final List<byte[]> result = new ArrayList<byte[]>( buffers.size() );         final long start2 = System.currentTimeMillis();         for ( final byte[] ar : encoded )             result.add( codec.decodeBytes(ar) );         final long decodeTime = System.currentTimeMillis() - start2;         for ( int i = 0; i < buffers.size(); ++i )         {             if ( !Arrays.equals( buffers.get( i ), result.get( i ) ) )                 System.out.println( "Diff at pos = " + i );         }         return new TestResult( encodeTime / 1000.0, decodeTime / 1000.0 );     }

测试结果

jvm参数:-Xms512m -Xmx4G

一切都很明显了,从上面看出,sun的表现不是很好,IHarder和MigBase64性能可以接受,传说MigBase64性能第一,那也是过去了,在这次测试结果中,新的java8 base64运行速度最好,javaXml表现次之。

总结

如果你需要一个性能好,可靠的Base64编解码器,不要找JDK外面的了,java8里面的java.util.Base64以及java6中隐藏很深的javax.xml.bind.DatatypeConverter,他们两个都是不错的选择。

本篇中所有代码都在http://git.oschina.net/benhail/javase8-sample ,欢迎大家去关注下载

标签:编解码,BASE64,String,Base64,base64,API,new,byte,final
From: https://www.cnblogs.com/dirgo/p/17904214.html

相关文章

  • Swagger文档生成艺术:掌握@ApiModel和@ApiModelProperty的用法
    在现代软件开发中,提供清晰全面的 API文档 至关重要。@ApiModel 和 @ApiModelProperty 这样的代码注解在此方面表现出色,通过增强模型及其属性的元数据来丰富文档内容。它们的主要功能是为这些元素命名和描述,使生成的API文档更加明确。@ApiModel 和 @ApiModelProperty 的......
  • EFCore 实体的配置FluentAPI(3)
    上一篇文章简述了如何使用EFCore的CodeFirst进行数据库创建并实现简单的增删改查,这章我们讨论一些复杂的配置:实体配置实体配置目的:生成想要的表格类型,如表名,表的列名,表字段类型,长度等,下图是上一章EFCore生成的表 实体配置遵循约定大于配置的原则:就是说EFCore这个ORM框架已经......
  • net8获取泛微token以及访问api示例
    工作中涉及到调用泛微的场景,官方的示例又臭又长,抽空用NET8简化了写法,为了简化http访问,用了Flurl.Http这个库。在座各位大佬,我们直接就看代码了 usingSystem.Security.Cryptography;usingSystem.Text.Json;usingFlurl.Http;//请按照官方内容做好初始化工作https://e-c......
  • ffmpeg 添加自定义编解码插件
    有两种方法:一.ffmpeg添加自定义编解码插件(以修改ffmpeg源码的方式添加)例:添加一个解码器,给这个解码器取个名字叫mydecoder,可以通过下面的步骤添加:1.在libavcodec目录下,新建文件mydecoder.c#include"avcodec.h"#include"codec_internal.h"//自己封装的编解码器的头文件#......
  • 【HarmonyOS】如何实现entry模块页面跳转HSP共享包页面(API9)
    【关键字】API9、ArkTS、跨包路由跳转、entry模块页面跳转HSP共享包页面 【写在前面】本篇文章主要介绍使用API9ArkTS开发鸿蒙应用时,如何实现从entry模块或feature模块页面,跳转到HSPlibrary模块的页面。 【开发步骤】第一步:在API9Stage模型的工程中,【右键-New-Module......
  • JavaScript-IndexedDB API
    概述随着浏览器的功能不断增强,越来越多的网站开始考虑,将大量数据储存在客户端,这样可以减少从服务器获取数据,直接从本地获取数据。现有的浏览器数据储存方案,都不适合储存大量数据:Cookie的大小不超过4KB,且每次请求都会发送回服务器;LocalStorage在2.5MB到10MB之间(各家浏览器不......
  • BAPI_ASSET_TRANSFER_POST Error
    1、BAPIACAM1X_MAX_DE-WAERS为初始值;货币金额1.000,00000000(AMOUNT格式)已传递,但无货币调用BAPI时,只传递了金额,没有传币种,补充币种。1LS_TRANSFERPOSTINGDATA-AMOUNT=1000.2LS_TRANSFERPOSTINGDATA-CURRENCY='CNY'.2、没有对资产000001000000-0000定义折旧范......
  • 抖音商品详情API接口在电商行业中的重要性及实时数据获取实现
    随着移动互联网的快速发展,电商行业不断壮大。抖音作为一款短视频社交应用,近年来在电商领域取得了显著成果。本文将探讨抖音商品详情API接口在电商行业中的重要性,以及如何通过实时数据获取提高业务效率。我们将介绍相关的技术背景、API接口的基础知识、实时数据获取的方法和代码实现......
  • 抖音关键词商品列表API:电商行业的成功之道及实时数据获取的魔力
    随着电商行业的快速发展,抖音作为一款短视频社交应用,已经成为电商领域的重要平台。抖音关键词商品列表API接口在电商行业中具有重要意义,它可以帮助商家快速获取商品信息,提高销售效率。本文将探讨抖音关键词商品列表API接口在电商行业中的重要性,以及如何通过实时数据获取提高业务效率......
  • 使用 FastEndpoints 来垂直切换Web API的控制器方法
    在我们开发项目的WebAPI的时候,随着项目功能要求越来越多,可能我们会为控制器基类增加越来越多的基础功能,有些功能有一定的适应性,但可能在一般的子类中用不到,而随着对控制器控制要求越来越精细,那么需要为基类或者子类增加更多的控制功能,这样随着迭代的进行,有些控制器的功能会显得越......