okHttp 解决文件上传中文名错误 Unexpected char

错误描述

使用 okHttp 调用接口进行文件上传,以下代码报错:Unexpected char 0x6d4b at 34 in Content-Disposition value: form-data

java 复制代码
RequestBody fileBody = RequestBody.create(MediaType.parse("application/octet-stream"), file);
MultipartBody.Part filePart = MultipartBody.Part.createFormData("file", fileName, fileBody);

解决方法

这个问题只在 OkHttp 3.12.0、3.12.1、3.12.2 版本中出现,应该是一个bug,后续版本解决了这个问题,所以你可以:

  1. 避免使用上述的 OkHttp 版本
  2. 手动生成 Headers ,代码如下:
java 复制代码
List<MultipartBody.Part> parts = new ArrayList<>();
// 添加文件参数
RequestBody fileBody = RequestBody.create(MediaType.parse("application/octet-stream"), file);
Headers fileHeaders = new Headers.Builder().addUnsafeNonAscii(
        "Content-Disposition","form-data; name=\"file\"; filename=\"" + fileName + "\"").build();
MultipartBody.Part filePart = MultipartBody.Part.create(fileHeaders , fileBody);
parts.add(filePart);

原因分析

报错的原因是OkHttp的Header检查机制不允许在Content-Disposition头中使用非ASCII字符,所以执行 MultipartBody.Part filePart = MultipartBody.Part.createFormData("file", fileName, fileBody); 这段代码时就会报错。

我们来查看源码分析一下 createFormData 方法: 调用了 Headers.of() 方法来生成 Headers ,再来查看 Headers.of()方法
这里执行了 checkValue(value, name); 方法,在这个方法里检查 value 中是否包含 非ASCII字符 。

解决过程

问题出在 MultipartBody.Part.createFormData() 方法,所以就不能调用这个方法来创建 MultipartBody.Part MultipartBody.Part.createFormData() 方法内部是调用MultipartBody.Part.create() 来创建 MultipartBody.Part

java 复制代码
return create(Headers.of(new String[]{"Content-Disposition", disposition.toString()}), body);

那么就可以改用 MultipartBody.Part.create() 来创建 MultipartBody.Part 此方法需要传入 Headers 对象,那么我们来查看Headers 对象需要怎么创建。 我留意到 Headers.Builder 中有这么一个方法 :addUnsafeNonAscii()

java 复制代码
public Builder addUnsafeNonAscii(String name, String value) {
    Headers.checkName(name);
    return this.addLenient(name, value);
}

调用此方法能够不对 Form 参数的值校验是否包含非 ASCII 字符,所以调用该方法来生成 Headers就能避免此问题。 将原先代码:

java 复制代码
RequestBody fileBody = RequestBody.create(MediaType.parse("application/octet-stream"), file);
MultipartBody.Part filePart = MultipartBody.Part.createFormData("file", fileName, fileBody);

改为

java 复制代码
RequestBody fileBody = RequestBody.create(MediaType.parse("application/octet-stream"), file);
Headers fileHeaders = new Headers.Builder().addUnsafeNonAscii(
        "Content-Disposition","form-data; name=\"file\"; filename=\"" + fileName + "\"").build();
MultipartBody.Part filePart = MultipartBody.Part.create(fileHeaders , fileBody);

将原先使用 MultipartBody.Part.createFormData() 方法生成 MultipartBody.Part ,改为使用 MultipartBody.Part.create 生成MultipartBody.Part,并且手动创建 Headers 对象。

参考

okhttp上传文件中文名报错问题分析。分析java.lang.IllegalArgumentException: Un - 掘金

相关推荐
北城以北88882 分钟前
SpringBoot--Spring Boot原生缓存基于Redis的Cacheable注解使用
java·spring boot·redis·缓存·intellij-idea
武子康2 分钟前
Java-208 RabbitMQ Topic 主题交换器详解:routingKey/bindingKey 通配符与 Java 示例
java·分布式·性能优化·消息队列·系统架构·rabbitmq·java-rabbitmq
后端小张1 小时前
【JAVA 进阶】SpringMVC全面解析:从入门到实战的核心知识点梳理
java·开发语言·spring boot·spring·spring cloud·java-ee·springmvc
Lucky小小吴2 小时前
ClamAV扫描速度提升6.5倍:服务器杀毒配置优化实战指南
java·服务器·网络·clamav
handsome_sai7 小时前
【Java 线程池】记录
java
大学生资源网8 小时前
基于springboot的唐史文化管理系统的设计与实现源码(java毕业设计源码+文档)
java·spring boot·课程设计
guslegend9 小时前
SpringSecurity源码剖析
java
roman_日积跬步-终至千里9 小时前
【人工智能导论】02-搜索-高级搜索策略探索篇:从约束满足到博弈搜索
java·前端·人工智能
大学生资源网10 小时前
java毕业设计之儿童福利院管理系统的设计与实现(源码+)
java·开发语言·spring boot·mysql·毕业设计·源码·课程设计