聊聊MultipartFile與File的一些事兒
前言
前段時間碰到了中轉文件的需求,需要使用HttpClient中轉一下文件,過程為:

在實現(xiàn)這個需求的過程中就用得到了MultipartFile與File,而且我對前一個也不是很熟悉。記錄一下
什么是MultipartFile
MultipartFile是spring類型,代表HTML中form data方式上傳的文件,包含二進制數(shù)據(jù)+文件名稱?!緛碜园俣戎馈?/p>
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());
// 創(chuàng)建文件
f = new File(multipartFile.getOriginalFilename());
try ( InputStream in = multipartFile.getInputStream(); OutputStream os = new FileOutputStream(f)){
// 得到文件流。以文件流的方式輸出到新文件
// 可以使用byte[] ss = multipartFile.getBytes();代替while
byte[] 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的URL
System.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只是一個空殼

你能獲取的也只有這個F:\upload_edfce39f_2894_4b66_b865_d5fb8636bdf3_00000000.tmp 網(wǎng)上有說會在根目錄生成臨時文件的,從tmp也可以看出來是個臨時文件,但是我試了好幾次啥都沒找到。。。。
直接獲取這個file讀取內(nèi)容也是會報文件找不到的 這是必然的 當然也有在spring配置文件配置CommonsMultipartResolver的 這就感覺很麻煩了。。。。
但是我們可以看一下diskFileItem 看下圖 是不是很熟悉了,從diskFileItem可以獲取文件流,其實你看了源碼你就知道獲取文件流都是從這里獲取的。剩下的就好辦了 我就不贅述了/。

在使用臨時文件的時候可以使用緩沖區(qū)創(chuàng)建臨時文件
// 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.txt
File file =File.createTempFile("tmp", ".txt");
HttpClient構建上傳文件參數(shù)并實現(xiàn)中轉文件
這里不自己給例子了,參考了其他博客的代碼
// 獲取文件名稱
String fileName = file.getOriginalFilename();
HttpPost httpPost = new HttpPost(url);
// 創(chuàng)建文件上傳實體
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);// 執(zhí)行提交
執(zhí)行提交之后你會發(fā)現(xiàn)你上傳的文件名會出現(xiàn)中文亂碼
這里參考
HttpClient上傳文件中文名亂碼 該文章詳細說明了為什么會亂碼以及怎么解決
我使用的解決辦法是:
//設置模式為RFC6532 builder.setMode(HttpMultipartMode.RFC6532);
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Java Swing實現(xiàn)坦克大戰(zhàn)游戲
這篇文章主要介紹了Java Swing實現(xiàn)坦克大戰(zhàn)游戲,文中有非常詳細的代碼示例,對正在學習java的小伙伴們有很大的幫助喲,需要的朋友可以參考下2021-05-05
使用springboot activiti關閉驗證自動部署方式
這篇文章主要介紹了使用springboot activiti關閉驗證自動部署方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09
lambda表達式與傳統(tǒng)接口函數(shù)實現(xiàn)方式對比詳解
這篇文章主要為大家介紹了lambda表達式與傳統(tǒng)接口函數(shù)實現(xiàn)方式對比詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家度偶多進步早日升職加薪2022-03-03
java?socket實現(xiàn)局域網(wǎng)聊天
這篇文章主要為大家詳細介紹了java?socket實現(xiàn)局域網(wǎng)聊天,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-05-05

