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


前言

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


提示:以下是本篇文章正文内容,下面案例可供参考

一、问题描述

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


二、解决方法

解决方法相对简单,只是将调用Base64解密的方式由getDecoder()替换成getMimeDecoder()后问题就得到了解决。
在这里插入图片描述

运行截图:
在这里插入图片描述


三、问题原因

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

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

当看到两种方式的区别后,好像意识到了问题所在,接着在IntelliJ IDEA中进行测试:
在测试时使用的Base64为JDK1.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);
    }
}

在这里将测试字符串使用两种编码方式进行编码,查看在控制台输出的内容是否相同
运行结果:
在这里插入图片描述
此时将测试字符串缩短长度以后,再次进行测试
在这里插入图片描述

对于编码结果来说,在加密后的字符长度不到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)));
    }
}

在这里插入图片描述
观察运行结果可以发现在加密后的字符长度小于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)));
    }
}

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

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

在这里插入图片描述


总结

经过测试发现两种编码方式进行加密和解密的区别以后,就又引出了一个问题,就是我在加密文件内容时是使用了基本编码方式进行加密,但是使用基本加密方式进行解密时却无法进行解密
在这里插入图片描述
这里是文件加密覆盖后的结果,但是当解密时基本的编码方式又不起作用了,后来找到了真正的原因:问题就在于我对文件进行解密时要先将文件内容读取到StringBuilder中这就是我们最后要进行Base64解码的最终内容,我在读取时增加了一个标志变量查看读取到的内容到底是多少行时,发现了事情的真相
在这里插入图片描述
现在再回顾为什么会出现基本的编码方式却解码不了基本方式编码后内容的问题,原因就在于我们Base64解码的内容实际是有两行,但是我们只能看到一行,因此也可以发现其实MIME的加密字符一行最多只能有76个字符,但是也可以解密一行超过76个字符长度的加密字符,同时JDK8中的Base64基本加密方式无法解码加密字符中含有换行的数据,哪怕第二行没有数据,只能有一行加密字符

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

  • 59
    点赞
  • 56
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 6
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Dream_飞翔

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值