聊聊MultipartFile与File的一些事儿
前言
前段时间碰到了中转文件的需求,需要使用HttpClient中转一下文件,过程为:
文章图片
在实现这个需求的过程中就用得到了MultipartFile与File,而且我对前一个也不是很熟悉。记录一下
什么是MultipartFile
MultipartFile是spring类型,代表HTML中form data方式上传的文件,包含二进制数据+文件名称。【来自百度知道】
MultipartFile 与 File 的 互相转换
1. File转MultipartFile
(1):使用org.springframework.mock.web.MockMultipartFile 需要导入spring-test.jar
public static void main(String[] args) throws Exception {String filePath = "F:\\test.txt"; File file = new File(filePath); FileInputStream fileInputStream = new FileInputStream(file); // MockMultipartFile(String name, @Nullable String originalFilename, @Nullable String contentType, InputStream contentStream)// 其中originalFilename,String contentType 旧名字,类型可为空// ContentType.APPLICATION_OCTET_STREAM.toString() 需要使用HttpClient的包MultipartFile multipartFile = new MockMultipartFile("copy"+file.getName(),file.getName(),ContentType.APPLICATION_OCTET_STREAM.toString(),fileInputStream); System.out.println(multipartFile.getName()); // 输出copytest.txt}
(2):使用CommonsMultipartFile
public static void main(String[] args) throws Exception {String filePath = "F:\\test.txt"; File file = new File(filePath); // 需要导入commons-fileupload的包FileItem fileItem = new DiskFileItem("copyfile.txt", Files.probeContentType(file.toPath()),false,file.getName(),(int)file.length(),file.getParentFile()); byte[] buffer = new byte[4096]; int n; try (InputStream inputStream = new FileInputStream(file); OutputStream os = fileItem.getOutputStream()){while ( (n = inputStream.read(buffer,0,4096)) != -1){os.write(buffer,0,n); }//也可以用IOUtils.copy(inputStream,os); MultipartFile multipartFile = new CommonsMultipartFile(fileItem); System.out.println(multipartFile.getName()); }catch (IOException e){e.printStackTrace(); }}
2. MultipartFile转File
(1):使用File转MultipartFile的逆过程
你在看这个代码的时候会觉得很熟悉,是的这个就是File转MultipartFile的逆转过程,这个方法会在根目录生成一个文件,需要删除该文件。
public static void main(String[] args) throws Exception {int n; // 得到MultipartFile文件MultipartFile multipartFile = getFile(); File f = null; // 输出文件的新name 就是指上传后的文件名称System.out.println("getName:"+multipartFile.getName()); // 输出源文件名称 就是指上传前的文件名称System.out.println("Oriname:"+multipartFile.getOriginalFilename()); // 创建文件f = new File(multipartFile.getOriginalFilename()); try ( InputStream in= multipartFile.getInputStream(); OutputStream os = new FileOutputStream(f)){// 得到文件流。以文件流的方式输出到新文件// 可以使用byte[] ss = multipartFile.getBytes(); 代替whilebyte[] buffer = new byte[4096]; while ((n = in.read(buffer,0,4096)) != -1){os.write(buffer,0,n); }// 读取文件第一行BufferedReader bufferedReader = new BufferedReader(new FileReader(f)); System.out.println(bufferedReader.readLine()); // 输出路径bufferedReader.close(); }catch (IOException e){e.printStackTrace(); }// 输出file的URLSystem.out.println(f.toURI().toURL().toString()); // 输出文件的绝对路径System.out.println(f.getAbsolutePath()); // 操作完上的文件 需要删除在根目录下生成的文件File file = new File(f.toURI()); if (file.delete()){System.out.println("删除成功"); }else {System.out.println("删除失败"); }}/**** @Description 返回MultipartFile文件* @return org.springframework.web.multipart.MultipartFile* @date 2019/1/5 11:08* @auther dell*/public static MultipartFile getFile() throws IOException {String filePath = "F:\\test.txt"; File file = new File(filePath); FileItem fileItem = new DiskFileItem("copyfile.txt", Files.probeContentType(file.toPath()),false,file.getName(),(int)file.length(),file.getParentFile()); byte[] buffer = new byte[4096]; int n; try (InputStream inputStream = new FileInputStream(file); OutputStream os = fileItem.getOutputStream()){while ( (n = inputStream.read(buffer,0,4096)) != -1){os.write(buffer,0,n); }//也可以用IOUtils.copy(inputStream,os); MultipartFile multipartFile = new CommonsMultipartFile(fileItem); System.out.println(multipartFile.getName()); return multipartFile; }catch (IOException e){e.printStackTrace(); }return null; }
(2):使用transferTo (本质上还是使用了流 只不过是封装了步骤)
会生成文件,最后不需要文件要删除
public static void main(String[] args) throws Exception {String path = "F:\\demo\\"; File file = new File(path,"demo.txt"); // 得到MultipartFile文件MultipartFile multipartFile = getFile(); /*byte[] ss = multipartFile.getBytes(); OutputStream os = new FileOutputStream(file); os.write(ss); os.close(); */multipartFile.transferTo(file); // 读取文件第一行BufferedReader bufferedReader = new BufferedReader(new FileReader(file)); System.out.println(bufferedReader.readLine()); // 输出绝对路径System.out.println(file.getAbsolutePath()); bufferedReader.close(); // 操作完上的文件 需要删除在根目录下生成的文件if (file.delete()){System.out.println("删除成功"); }else {System.out.println("删除失败"); }}/**** @Description 返回MultipartFile文件* @return org.springframework.web.multipart.MultipartFile* @date 2019/1/5 11:08* @auther dell*/public static MultipartFile getFile() throws IOException {String filePath = "F:\\test.txt"; File file = new File(filePath); FileItem fileItem = new DiskFileItem("copyfile.txt", Files.probeContentType(file.toPath()),false,file.getName(),(int)file.length(),file.getParentFile()); byte[] buffer = new byte[4096]; int n; try (InputStream inputStream = new FileInputStream(file); OutputStream os = fileItem.getOutputStream()){while ( (n = inputStream.read(buffer,0,4096)) != -1){os.write(buffer,0,n); }//也可以用IOUtils.copy(inputStream,os); MultipartFile multipartFile = new CommonsMultipartFile(fileItem); System.out.println(multipartFile.getName()); return multipartFile; }catch (IOException e){e.printStackTrace(); }return null; }
(3):使用FileUtils.copyInputStreamToFile()
也是会生成文件,到最后也是要删除文件
public static void main(String[] args) throws Exception {String path = "F:\\demo\\"; File file = new File(path,"demo.txt"); // 得到MultipartFile文件MultipartFile multipartFile = getFile(); // 把流输出到文件FileUtils.copyInputStreamToFile(multipartFile.getInputStream(),file); // 读取文件第一行BufferedReader bufferedReader = new BufferedReader(new FileReader(file)); System.out.println(bufferedReader.readLine()); // 输出绝对路径System.out.println(file.getAbsolutePath()); bufferedReader.close(); // 操作完上的文件 需要删除在根目录下生成的文件if (file.delete()){System.out.println("删除成功"); }else {System.out.println("删除失败"); }}/**** @Description 返回MultipartFile文件* @return org.springframework.web.multipart.MultipartFile* @date 2019/1/5 11:08* @auther dell*/public static MultipartFile getFile() throws IOException {String filePath = "F:\\test.txt"; File file = new File(filePath); FileItem fileItem = new DiskFileItem("copyfile.txt", Files.probeContentType(file.toPath()),false,file.getName(),(int)file.length(),file.getParentFile()); byte[] buffer = new byte[4096]; int n; try (InputStream inputStream = new FileInputStream(file); OutputStream os = fileItem.getOutputStream()){while ( (n = inputStream.read(buffer,0,4096)) != -1){os.write(buffer,0,n); }//也可以用IOUtils.copy(inputStream,os); MultipartFile multipartFile = new CommonsMultipartFile(fileItem); System.out.println(multipartFile.getName()); return multipartFile; }catch (IOException e){e.printStackTrace(); }return null; }
3:强转类型
CommonsMultipartFile commonsmultipartfile = (CommonsMultipartFile) multipartFile; DiskFileItem diskFileItem = (DiskFileItem) commonsmultipartfile.getFileItem(); File file = diskFileItem.getStoreLocation();
这种强转你获得的file只是一个空壳
【聊聊MultipartFile与File的一些事儿】
文章图片
你能获取的也只有这个F:\upload_edfce39f_2894_4b66_b865_d5fb8636bdf3_00000000.tmp 网上有说会在根目录生成临时文件的,从tmp也可以看出来是个临时文件,但是我试了好几次啥都没找到。。。。
直接获取这个file读取内容也是会报文件找不到的 这是必然的 当然也有在spring配置文件配置CommonsMultipartResolver的 这就感觉很麻烦了。。。。
但是我们可以看一下diskFileItem 看下图 是不是很熟悉了,从diskFileItem可以获取文件流,其实你看了源码你就知道获取文件流都是从这里获取的。剩下的就好办了 我就不赘述了/。
文章图片
在使用临时文件的时候可以使用缓冲区创建临时文件
//createTempFile(String prefix, String suffix) //prefix 文件名 suffix 文件格式// 默认是tmp格式C:\Users\dell\AppData\Local\Temp\tmp8784723057512789016.tmp File file =File.createTempFile("tmp", null); // txt格式C:\Users\dell\AppData\Local\Temp\tmp2888293586594052933.txtFile file =File.createTempFile("tmp", ".txt");
HttpClient构建上传文件参数并实现中转文件
这里不自己给例子了,参考了其他博客的代码
// 获取文件名称String fileName = file.getOriginalFilename(); HttpPost httpPost = new HttpPost(url); // 创建文件上传实体MultipartEntityBuilder builder = MultipartEntityBuilder.create(); builder.addBinaryBody("file", file.getInputStream(), ContentType.MULTIPART_FORM_DATA, fileName); builder.addTextBody("filename", fileName); HttpEntity entity = builder.build(); httpPost.setEntity(entity); HttpResponse response = httpClient.execute(httpPost); // 执行提交
执行提交之后你会发现你上传的文件名会出现中文乱码
这里参考
HttpClient上传文件中文名乱码 该文章详细说明了为什么会乱码以及怎么解决
我使用的解决办法是:
//设置模式为RFC6532 builder.setMode(HttpMultipartMode.RFC6532);
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
推荐阅读
- JAVA(抽象类与接口的区别&重载与重写&内存泄漏)
- Docker应用:容器间通信与Mariadb数据库主从复制
- 《真与假的困惑》???|《真与假的困惑》??? ——致良知是一种伟大的力量
- 第326天
- Shell-Bash变量与运算符
- 逻辑回归的理解与python示例
- Guava|Guava RateLimiter与限流算法
- 我和你之前距离
- CGI,FastCGI,PHP-CGI与PHP-FPM
- 原生家庭之痛与超越