使用HttpClient MultipartEntityBuilder 上傳文件,並解決中文文件名亂碼問題

来源:https://www.cnblogs.com/evasean/archive/2018/07/25/9368670.html
-Advertisement-
Play Games

遇到一種業務場景,前端上傳的文件需要經過java服務轉發至文件服務。期間遇到了原生HttpClient怎麼使用的問題、怎麼把MultipartFile怎麼重新組裝成Http請求發送出去的問題、文件中文名亂碼問題。最後都解決了,先上代碼,再講遇到的坑 特別說明及遇到的坑: 1. 這裡基於tomcat進 ...


遇到一種業務場景,前端上傳的文件需要經過java服務轉發至文件服務。期間遇到了原生HttpClient怎麼使用的問題、怎麼把MultipartFile怎麼重新組裝成Http請求發送出去的問題、文件中文名亂碼問題。最後都解決了,先上代碼,再講遇到的坑

 1 @Slf4j
 2 @Service
 3 public class FileServiceImpl implements IFileService {
 4 
 5     @Value("${FileService.putUrl}")
 6     private String putUrl;
 7     @Value("${FileService.app_id}")
 8     private String appId;
 9     @Value("${FileService.securityKey}")
10     private String secureKey;
11 
12     private final static String UPLOAD_RESPONSE_CODE = "error";
13     private final static Integer UPLOAD_RESPONSE_SUCCESS = 0;
14 
15 
16     @Override
17     public String upload(MultipartFile file) {
18 
19         int timeOut = 30000;
20         CloseableHttpClient httpClient = HttpClientBuilder.create().build();
21         HttpHost proxy = new HttpHost("127.0.0.1", 62145, "http"); //設置本地fiddler代理,方便排查問題
22         RequestConfig requestConfig = RequestConfig.custom().setConnectionRequestTimeout(timeOut)
23                 .setConnectTimeout(timeOut).setSocketTimeout(timeOut).setProxy(proxy).build();
24         HttpPost httpPost = new HttpPost(putUrl);
25         httpPost.setConfig(requestConfig);
26         try {
27             //BROWSER_COMPATIBLE自定義charset,RFC6532=utf-8,STRICT=iso-8859-1
28             //此處一定要用RFC6532,網上普遍用的BROWSER_COMPATIBLE依然會出現中文名亂碼
29             MultipartEntityBuilder multipartEntityBuilder = MultipartEntityBuilder.create().setMode(HttpMultipartMode.RFC6532);
30             //multipartEntityBuilder.setCharset(Charset.forName("UTF-8")); //此處踩坑,轉發出去的filename依然為亂碼
31            //ContentType contentType = ContentType.create("multipart/form-data",Charset.forName("UTF-8")); //此處也是坑,轉發出去的filename依然為亂碼
32             multipartEntityBuilder.addBinaryBody("file", file.getInputStream(), ContentType.DEFAULT_BINARY, file.getOriginalFilename());
33 
34             multipartEntityBuilder.addTextBody("app_id", appId); //可替換成你自己需要的附加欄位
35             long time = System.currentTimeMillis() / 1000;
36             multipartEntityBuilder.addTextBody("time", String.valueOf(time)); //可替換成你自己需要的附加欄位
37             String beforeSign = String.format("app_id=%s&time=%s%s", appId, time, secureKey);
38             String sign = MD5Util.md5(beforeSign);
39             multipartEntityBuilder.addTextBody("sign", sign); //可替換成你自己需要的附加欄位
40 
41             HttpEntity requestEntity = multipartEntityBuilder.build();
42             httpPost.setEntity(requestEntity);
43             HttpResponse httpResponse = httpClient.execute(httpPost);
44             int statusCode= httpResponse.getStatusLine().getStatusCode();
45             if (statusCode != 200) throw new BizException(BizCode.INNER_SERVICE_ERROR, "響應狀態碼為:" + statusCode);
46             HttpEntity responseEntity = httpResponse.getEntity();
47             return getUrlString(EntityUtils.toString(responseEntity));
48         } catch (Exception e) {
49             log.error("發送文件異常:{}", e);
50             throw new BizException(BizCode.INNER_SERVICE_ERROR, "發送文件服務異常:" + e.getMessage());
51         } finally {
52             try {
53                 httpClient.close();
54             } catch (IOException e) {
55                 log.error("關閉httpClient異常:" + e.getMessage(), e);
56             }
57         }
58     }
59 
60     private String getUrlString(String jsonString) {
61         try{
62             log.debug("解析json串:"+ jsonString);
63             JSONObject jsonObject = JSONObject.parseObject(jsonString);
64             if (jsonObject.getInteger(UPLOAD_RESPONSE_CODE) != UPLOAD_RESPONSE_SUCCESS) {
65                 log.error("文件服務返回錯誤:" + jsonObject.getString("data"));
66                 throw new OtherServiceReturnErrorException("文件服務返回錯誤:" + jsonObject.getString("data"));
67             }
68             return ((JSONObject)jsonObject.get("data")).get("original").toString();
69         }catch (Exception e) {
70             log.error("文件服務返回json解析錯誤:" + jsonString);
71             throw new OtherServiceReturnErrorException("文件服務返回json解析錯誤:" + jsonString);
72         }
73 
74     }
75 }

特別說明及遇到的坑:

1. 這裡基於tomcat進行請求轉發,需要在代碼中手動添加代理:

HttpHost proxy = new HttpHost("127.0.0.1", 62145, "http"); //設置本地fiddler代理,方便排查問題
RequestConfig requestConfig = RequestConfig.custom().setConnectionRequestTimeout(timeOut)
.setConnectTimeout(timeOut).setSocketTimeout(timeOut).setProxy(proxy).build();

2. MultipartFile通過getInputStream()可以將流設置到MultipartEntityBuilder中,其中addBinaryBody裡面的ContentType 和 filename必須設置,要不然後續服務讀取不到這個文件流

MultipartEntityBuilder multipartEntityBuilder = MultipartEntityBuilder.create().setMode(HttpMultipartMode.RFC6532);
multipartEntityBuilder.addBinaryBody("file", file.getInputStream(), ContentType.DEFAULT_BINARY, file.getOriginalFilename());

3. file.getOriginalFilename()方法雖然沒有亂碼,但是addBinaryBody後,組裝的Http請求出去總是亂碼,如下圖:

 

踩了各種坑,如為MultipartEntityBuilder設置Charset或者是手動設置ContentType,都無法解決此問題,文件名依然是上圖所示亂碼

後來發現在MultipartEntityBuilder中設置Mode為HttpMultipartMode.RFC6532可以完美解決這個問題,並且不再需要單獨設置ContentType或Charset,因為HttpMultipartMode.RFC6532就告訴了MultipartEntityBuilder,裡面的數據都要使用UTF-8進行處理,fiddler抓到的請求發現filename成功變成中文名。

 

 

done



您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、什麼是This逃逸? 在構造器構造還未徹底完成前(即實例初始化階段還未完成),將自身this引用向外拋出並被其他線程複製(訪問)了該引用,可能會問到該還未被初始化的變數,甚至可能會造成更大嚴重的問題。 廢話不多說,看一下代碼 輸出結果:這說明ThisEscape還未完成實例化,構造還未徹底結束。 ...
  • 註:本文是以MASM的語法格式為基礎的,大部分內容參考《Intel彙編語言程式設計 第五版》 1、標識符 標識符是程式員自己定義的名字,用來標識變數、常量、過程或代碼標號。創建標識符時要註意: 1)標識符可以包含1~247個字元; 2)標識符大小寫不敏感(MASM預設不敏感); 3)標識符的第一個字 ...
  • 查看String的源碼可以發現它以一個char類型的數組保存字元串的,而String.length()方法返回的也是這個char數組的長度. 那麼,這個長度和"字元"長度有什麼關係呢? 在這裡就不得不引入碼點和代碼單元的概念,以下是摘抄至《Java核心技術捲一基礎知識(第十版)》中的定義: "碼點( ...
  • 實現HandlerInterceptor介面或者繼承HandlerInterceptor的子類,比如Spring 已經提供的實現了HandlerInterceptor 介面的抽象類HandlerInterceptorAdapter ,下麵講實現其介面的寫法,先看一下這個介面的三個方法. - 方法pr ...
  • Python中,主要的基本類型有:數字(int型)、字元串(string型)、列表(list型)、元祖(tuple型)、字典(direct型)、布爾值(boolean型) 1.int型 1.強轉int 2.查看類型 3.轉換進位 2.string型 1.常見的 2.可進行格式替換 3.判斷類型 4. ...
  • 定義視圖 本質就是一個函數 視圖的參數 一個HttpRequest實例 通過正則表達式組獲取的位置參數 通過正則表達式組獲得的關鍵字參數 在應用目錄下預設有views.py文件,一般視圖都定義在這個文件中 如果處理功能過多,可以將函數定義到不同的py文件中 一個HttpRequest實例 通過正則表 ...
  • 1、概念 裝飾器(decorator)就是:定義了一個函數,想在運行時動態增加功能,又不想改動函數本身的代碼。可以起到復用代碼的功能,避免每個函數重覆性編寫代碼,簡言之就是拓展原來函數功能的一種函數。在python中,裝飾器(decorator)分為函數裝飾器和類裝飾器兩種。python中內置的@語 ...
  • OpenFlow1.3.3 學習記錄(持續更新) 正在學習OpenFlow1.3,該篇筆記將日常更新,主要內容大致為官方文檔的總結與翻譯。 交換機組件 按照優先順序順序進行包匹配,如果匹配到流表項,則執行流表項中綁定的Instructions;如果沒有匹配到流表項,將根據table miss的配置進行 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...