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

简介: 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


相关文章
|
3月前
|
XML Java 数据库连接
解码Java SPI:深入理解与实践【七】
解码Java SPI:深入理解与实践【七】
27 0
|
4月前
|
存储 Java Android开发
IO流:java中解码和编码出现乱码说明及代码实现
IO流:java中解码和编码出现乱码说明及代码实现
|
5月前
|
Java
学习一下Java中Character类的使用
学习一下Java中Character类的使用
28 0
|
6月前
|
Java
解决异常 java.net.URISyntaxException: Illegal character in query at index
解决异常 java.net.URISyntaxException: Illegal character in query at index
211 0
|
14天前
|
Java 编译器
Java Character 类
4月更文挑战第13天
|
19天前
|
监控 安全 Java
Java并发初探:解码多线程编程的基础
Java并发初探:解码多线程编程的基础
9 0
Java并发初探:解码多线程编程的基础
|
19天前
|
存储 监控 安全
泛型魔法:解码Java中的类型参数
泛型魔法:解码Java中的类型参数
35 0
泛型魔法:解码Java中的类型参数
|
29天前
|
安全 Java
java.security.InvalidKeyException: Illegal key size
java.security.InvalidKeyException: Illegal key size
9 0
|
2月前
java.lang.IllegalArgumentException: Invalid character found in method name
java.lang.IllegalArgumentException: Invalid character found in method name
18 0
|
2月前
|
Java
Java  Character 类
Java  Character 类
7 0