Base64解码遇到java.lang.IllegalArgumentException: Illegal base64 character d

本文涉及的产品
密钥管理服务KMS,1000个密钥,100个凭据,1个月
简介: Base64解码遇到java.lang.IllegalArgumentException: Illegal base64 character d

前言


在实现了将文件通过Base64的方式加密存储到数据库中并且读取到相应的文件时,在通过Base64的解密方法进行解密时,出现了不应该出现的错误,将解决问题的过程在这里进行记录和总结

一、问题描述


image.png

在这里进行解密时报了说d是Base64的非法字符的问题,但是在之前测试加密和解密时都没有报出这样的问题,因此在想出现问题的原因一定不是因为d是非法字符

二、解决方法


解决方法相对简单,只是将调用Base64解密的方式由getDecoder()替换成getMimeDecoder()后问题就得到了解决。

image.png

运行截图:

image.png

三、问题原因


在解决完问题之后再回过头来看为什么会出现这样的问题,我只是将项目中的解码方式换成了使用MIME型的base64编码方案就没有了解码错误的问题,而原本的Base64解码方式只是使用基本型的base64编码方案,两者之间到底有什么区别,在查阅资料后发现两种编码方式有以下区别:

参考资料:菜鸟教程_Java8 Base64

  1. 基本方式:输出被映射到一组字符A-Za-z0-9+/,编码不添加任何行标,输出的解码仅支持A-Za-z0-9+/。
  2. MIME方式:输出隐射到MIME友好格式。输出每行不超过76字符,并且使用’\r’并跟随’\n’作为分割。编码输出最后没有行分割。

当看到两种方式的区别后,好像意识到了问题所在,接着在IntelliJ IDEA中进行测试:

在测试时使用的Base64JDK1.8中自带的Base64工具类

在测试时先定义一个String类型的对象,接着分别使用两种编码方式进行加密

首先执行以下代码:

import java.nio.charset.StandardCharsets;
import java.util.Base64;
/**
 * @author Dream_飞翔
 * @date 2021/11/12
 * @time 22:27
 * @email 1072876976@qq.com
 */
public class TestBase64 {
    public static void main(String[] args) {
        String str01 = "这是一个测试两种编码方式的字符串";
        // 声明Base64基本方式编码对象
        Base64.Encoder encoderBase = Base64.getEncoder();
        // 声明Base64的MIME编码方式对象
        Base64.Encoder encoderMIME = Base64.getMimeEncoder();
        // 将同一字符以基本方式和MIME方式进行编码
        String encodeBase = encoderBase.encodeToString(str01.getBytes(StandardCharsets.UTF_8));
        String encodeMIME = encoderMIME.encodeToString(str01.getBytes(StandardCharsets.UTF_8));
        // 以UTF-8的格式编码字符串
        System.out.println("基本编码方式:\n" + encodeBase);
        System.out.println("MIME编码方式:\n" + encodeMIME);
    }
}

在这里将测试字符串使用两种编码方式进行编码,查看在控制台输出的内容是否相同

运行结果:

image.png

image.png

对于编码结果来说,在加密后的字符长度不到76位时输出结果完全相同,测试两种加密方式是否可以相互解密

测试代码

public class TestBase64 {
    public static void main(String[] args) {
        String str01 = "这是一个测试两种编码方式的字符串";
        // 声明Base64基本方式编码对象
        Base64.Encoder encoderBase = Base64.getEncoder();
        // 声明Base64的MIME编码方式对象
        Base64.Encoder encoderMIME = Base64.getMimeEncoder();
        // 将同一字符以基本方式和MIME方式进行编码
        String encodeBase = encoderBase.encodeToString(str01.getBytes(StandardCharsets.UTF_8));
        String encodeMIME = encoderMIME.encodeToString(str01.getBytes(StandardCharsets.UTF_8));
        // 以UTF-8的格式编码字符串
        System.out.println("基本编码方式:\n" + encodeBase);
        System.out.println("MIME编码方式:\n" + encodeMIME);
        // 进行解码
        Base64.Decoder decoderBase = Base64.getDecoder();
        Base64.Decoder decoderMIME = Base64.getMimeDecoder();
        // 将解密后的结果在控制台输出
        System.out.println("基本编码方式加密后的内容使用MIME编码方式的解密结果:\n" + new String(decoderMIME.decode(encodeBase)));
        System.out.println("MIME编码方式加密后的内容使用基本编码方式的解密结果:\n" + new String(decoderBase.decode(encodeMIME)));
    }
}

image.png

观察运行结果可以发现在加密后的字符长度小于76位时,两种编码方式可以对加密的内容相互进行解密

接着测试加密后的字符长度大于76位时,两种加密方式是否仍然可以相互解密

执行以下代码在控制台生成一个很长的文本,然后复制到字符串当中(手懒,不想手打数据了)

for (int i = 0;i < 200;i++) {
  System.out.print("测试");
}

测试代码

import java.nio.charset.StandardCharsets;
import java.util.Base64;
/**
 * @author Dream_飞翔
 * @date 2021/11/12
 * @time 00:27
 * @email 1072876976@qq.com
 */
public class TestBase64 {
    public static void main(String[] args) {
        String str01 = "测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试测试";
        // 声明Base64基本方式编码对象
        Base64.Encoder encoderBase = Base64.getEncoder();
        // 声明Base64的MIME编码方式对象
        Base64.Encoder encoderMIME = Base64.getMimeEncoder();
        // 将同一字符以基本方式和MIME方式进行编码
        String encodeBase = encoderBase.encodeToString(str01.getBytes(StandardCharsets.UTF_8));
        String encodeMIME = encoderMIME.encodeToString(str01.getBytes(StandardCharsets.UTF_8));
        // 以UTF-8的格式编码字符串
        System.out.println("基本编码方式:\n" + encodeBase);
        System.out.println("MIME编码方式:\n" + encodeMIME);
        // 进行解码
        Base64.Decoder decoderBase = Base64.getDecoder();
        Base64.Decoder decoderMIME = Base64.getMimeDecoder();
        // 将解密后的结果在控制台输出
        System.out.println("基本编码方式加密后的内容使用MIME编码方式的解密结果:\n" + new String(decoderMIME.decode(encodeBase)));
        System.out.println("MIME编码方式加密后的内容使用基本编码方式的解密结果:\n" + new String(decoderBase.decode(encodeMIME)));
    }
}

运行结果

image.png

对比到这里就不难理解,当使用Base64的基本编码方式进行解密时是无法解密加密字符中含有换行的内容,但是MIME的编码方式却可以无视要解密的字符是否有换行的情况,至于MIME编码方式是否是说的这样,现在进行测试。

最后验证是否可以使用各自的编码方式解密各自的加密内容

image.png

总结


经过测试发现两种编码方式进行加密和解密的区别以后,就又引出了一个问题,就是我在加密文件内容时是使用了基本编码方式进行加密,但是使用基本加密方式进行解密时却无法进行解密

image.png

这里是文件加密覆盖后的结果,但是当解密时基本的编码方式又不起作用了,后来找到了真正的原因:问题就在于我对文件进行解密时要先将文件内容读取到StringBuilder中,这就是我们最后要进行Base64解码的最终内容,我在读取时增加了一个标志变量查看读取到的内容到底是多少行时,发现了事情的真相

image.png

现在再回顾为什么会出现基本的编码方式却解码不了基本方式编码后内容的问题,原因就在于我们Base64解码的内容实际是有两行,但是我们只能看到一行,因此也可以发现其实MIME的加密字符一行最多只能有76个字符,但是也可以解密一行超过76个字符长度的加密字符,同时JDK8中的Base64基本加密方式无法解码加密字符中含有换行的数据,哪怕第二行没有数据,只能有一行加密字符。

有很多事是因为不想麻烦别人,所以自己咬咬牙就撑了过去。也有很多难以启齿的困难被自己死扛了过去。低潮期受到挫败的时候觉得自己没法振作了,最后还不是熬了过来。你看,我们都比自己想象中还要坚强。对待生活中的每一天若都像生命中的最后一天去对待,人生定会更精彩!

image.png


相关文章
|
4月前
|
Java 编译器 ice
【Java开发指南 | 第十五篇】Java Character 类、String 类
【Java开发指南 | 第十五篇】Java Character 类、String 类
50 1
|
4月前
|
存储 Java
java用base64编码案例
Java Base64编码示例:导入`java.util.Base64`,设置字符串`originalString`,使用`Base64.getEncoder().encodeToString()`编码并存储到`encodedString`,打印编码后字符串。解码用`Base64.getDecoder().decode()`。
47 0
|
2天前
|
Java 开发者
Java Character 类详解
Java中的`Character`类是`java.lang`包的一部分,用于将基本类型`char`封装为对象,并提供了丰富的静态方法来处理字符,如类型判断、大小写转换等。
|
6天前
|
Java
Java Character 类详解
`Character` 类是 Java 中的一个封装类,位于 `java.lang` 包中,主要用于处理单个字符。它是一个最终类,提供了多种静态方法来检查和操作字符属性,如判断字符是否为字母、数字或空格,以及转换字符的大小写等。此外,`Character` 类还支持自动装箱和拆箱,简化了 `char` 和 `Character` 之间的转换。以下是一些示例代码,展示了如何使用 `Character` 类的方法来检查字符属性和执行字符转换。掌握 `Character` 类的用法有助于更高效地处理字符数据。
|
1月前
|
存储 安全 Java
"Java编码魔法:揭秘图片与文件的Base64神秘转换术,让数据在指尖跳跃!"
【8月更文挑战第16天】Base64编码在Java开发中常用于将二进制数据如图片转换为ASCII字符串以便传输。编码使用64个字符及等号填充,每3字节数据编码为4个字符。Java利用`java.util.Base64`类实现此功能:读取图片或文件为字节数组后进行编码。解码时将Base64字符串还原为字节数组并写入文件。需注意编码效率降低、不提供安全性及特殊字符兼容性等问题。掌握这些技巧有助于解决Web开发中的数据传输需求。
53 4
|
1月前
|
Java
Java 图片、文件 Base64 互转
Java 图片、文件 Base64 互转
29 0
|
3月前
|
存储 Java 数据处理
Java的Base64加密解密详解
Java的Base64加密解密详解
|
3月前
|
Java
【技术解码】Java线程的五味人生:新建、就绪、运行、阻塞与死亡的哲学解读!
【6月更文挑战第19天】Java线程生命周期如同人生旅程,经历新建、就绪、运行、阻塞至死亡五阶段。从`new Thread()`的诞生到`start()`的蓄势待发,再到`run()`的全力以赴,线程在代码中奔跑。阻塞时面临挑战,等待资源释放,最终通过`join()`或中断结束生命。线程的每个状态转变,都是编程世界与哲思的交汇点。
27 1
|
3月前
|
编解码 数据可视化 Java
Java如何进行Base64的编码(Encode)与解码(Decode)?
Java如何进行Base64的编码(Encode)与解码(Decode)?
102 1
|
3月前
|
算法 Java API
Base64编码介绍及基于Java语言实现
Base64编码介绍及基于Java语言实现
26 0