springboot對(duì)壓縮請(qǐng)求的處理方法
springboot對(duì)壓縮請(qǐng)求的處理
最近對(duì)接銀聯(lián)需求,為了節(jié)省帶寬,需要對(duì)報(bào)文進(jìn)行壓縮處理。但是使用springboot自帶的壓縮設(shè)置不起作用:
server.compression.enabled=true server.compression.mime-types=application/javascript,text/css,application/json,application/xml,text/html,text/xml,text/plain server.compression.compressionMinSize=10
server.compression.enabled:表示是否開啟壓縮,默認(rèn)開啟,true:開啟,false:不開啟
server.compression.mime-types:壓縮的內(nèi)容的類型,有xml,json,html等格式
server.compression.compressionMinSize:開啟壓縮數(shù)據(jù)最小長度,單位為字節(jié),默認(rèn)是2048字節(jié)
源碼如下:
public class Compression {
private boolean enabled = false;
private String[] mimeTypes = new String[]{"text/html", "text/xml", "text/plain", "text/css", "text/javascript", "application/javascript", "application/json", "application/xml"};
private String[] excludedUserAgents = null;
private int minResponseSize = 2048;
}一、Tomcat設(shè)置壓縮原理
tomcat壓縮是對(duì)響應(yīng)報(bào)文進(jìn)行壓縮,當(dāng)請(qǐng)求頭中存在accept-encoding時(shí),如果Tomcat設(shè)置了壓縮,則會(huì)在響應(yīng)時(shí)對(duì)數(shù)據(jù)進(jìn)行壓縮。
tomcat壓縮源碼是在Http11Processor中設(shè)置的:
public class Http11Processor extends AbstractProcessor {
private boolean useCompression() {
// Check if browser support gzip encoding
MessageBytes acceptEncodingMB =
request.getMimeHeaders().getValue("accept-encoding");
if ((acceptEncodingMB == null)-->當(dāng)請(qǐng)求頭沒有這個(gè)字段是不進(jìn)行壓縮
|| (acceptEncodingMB.indexOf("gzip") == -1)) {
return false;
}
// If force mode, always compress (test purposes only)
if (compressionLevel == 2) {
return true;
}
// Check for incompatible Browser
if (noCompressionUserAgents != null) {
MessageBytes userAgentValueMB =
request.getMimeHeaders().getValue("user-agent");
if(userAgentValueMB != null) {
String userAgentValue = userAgentValueMB.toString();
if (noCompressionUserAgents.matcher(userAgentValue).matches()) {
return false;
}
}
}
return true;
}
}二、銀聯(lián)報(bào)文壓縮
作為發(fā)卡機(jī)構(gòu),銀聯(lián)報(bào)文請(qǐng)求報(bào)文是壓縮的,且報(bào)文頭中不存在accept-encoding字段,無法直接使用tomcat配置進(jìn)行壓縮解壓。
需要單獨(dú)處理這種請(qǐng)求
@RestController
@RequestMapping("/user")
public class UserController {
private static final Logger logger = LoggerFactory.getLogger(UserController.class);
/**
*
* application/xml格式報(bào)文
* */
@PostMapping(path = "/test", produces = MediaType.APPLICATION_XML_VALUE, consumes = MediaType.APPLICATION_XML_VALUE)
public void getUserInfoById(HttpServletRequest request, HttpServletResponse response) throws IOException {
String requestBody;
String resultBody="hello,wolrd";
byte[] returnByte;
if (StringUtils.isNoneEmpty(request.getHeader("Content-encoding"))) {
logger.info("報(bào)文壓縮,需要進(jìn)行解壓");
//業(yè)務(wù)處理
//返回報(bào)文也同樣需要進(jìn)行壓縮處理
assemleResponse(request,response,resultBody);
}
}
public static void assemleResponse(HttpServletRequest request, HttpServletResponse response,String resultBody) throws IOException {
response.setHeader("Content-Type","application/xml;charset=UTF-8");
response.setHeader("Content-Encoding","gzip");
byte[] returnByte=GzipUtil.compress(resultBody);
OutputStream outputStream=response.getOutputStream();
outputStream.write(returnByte);
}
}public class GzipUtil {
public static String uncompress(byte[] bytes){
if (bytes == null || bytes.length == 0) {
return null;
}
String requestBody=null;
ByteArrayInputStream byteArrayInputStream=new ByteArrayInputStream(bytes);
GZIPInputStream gzipInputStream = null;
try {
gzipInputStream = new GZIPInputStream(byteArrayInputStream);
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int temp;
while ((temp = gzipInputStream.read(buffer)) != -1) {
byteArrayOutputStream.write(buffer, 0, temp);
}
requestBody = new String(byteArrayOutputStream.toByteArray(), StandardCharsets.UTF_8);
} catch (IOException e) {
e.printStackTrace();
}
return requestBody;
}
public static String uncompress(HttpServletRequest request){
String requestBody=null;
int length = request.getContentLength();
try {
BufferedInputStream bufferedInputStream = new BufferedInputStream(request.getInputStream());
GZIPInputStream gzipInputStream = new GZIPInputStream(bufferedInputStream);
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int temp;
while ((temp = gzipInputStream.read(buffer)) != -1) {
byteArrayOutputStream.write(buffer, 0, temp);
}
requestBody = new String(byteArrayOutputStream.toByteArray(), StandardCharsets.UTF_8);
} catch (IOException e) {
e.printStackTrace();
}
return requestBody;
}
public static byte[] compress(String src){
if (src == null || src.length() == 0) {
return null;
}
ByteArrayOutputStream byteArrayOutputStream=new ByteArrayOutputStream();
try {
GZIPOutputStream gzipOutputStream=new GZIPOutputStream(byteArrayOutputStream);
gzipOutputStream.write(src.getBytes(StandardCharsets.UTF_8));
gzipOutputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
byte[] bytes=byteArrayOutputStream.toByteArray();
return bytes;
}
}補(bǔ)充:java springbooot使用gzip壓縮字符串
import lombok.extern.slf4j.Slf4j;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.util.zip.GZIPInputStream;
import java.util.zip.GZIPOutputStream;
/**
* effect:壓縮/解壓 字符串
*/
@Slf4j
public class CompressUtils {
/**
* effect 使用gzip壓縮字符串
* @param str 要壓縮的字符串
* @return
*/
public static String compress(String str) {
if (str == null || str.length() == 0) {
return str;
}
ByteArrayOutputStream out = new ByteArrayOutputStream();
GZIPOutputStream gzip = null;
try {
gzip = new GZIPOutputStream(out);
gzip.write(str.getBytes());
} catch (IOException e) {
log.error("",e);
} finally {
if (gzip != null) {
try {
gzip.close();
} catch (IOException e) {
log.error("",e);
}
}
}
return new sun.misc.BASE64Encoder().encode(out.toByteArray());
// return str;
}
/**
* effect 使用gzip解壓縮
*
* @param str 壓縮字符串
* @return
*/
public static String uncompress(String str) {
if (str == null) {
return null;
}
ByteArrayOutputStream out = new ByteArrayOutputStream();
ByteArrayInputStream in = null;
GZIPInputStream ginzip = null;
byte[] compressed = null;
String decompressed = null;
try {
compressed = new sun.misc.BASE64Decoder().decodeBuffer(str);
in = new ByteArrayInputStream(compressed);
ginzip = new GZIPInputStream(in);
byte[] buffer = new byte[1024];
int offset = -1;
while ((offset = ginzip.read(buffer)) != -1) {
out.write(buffer, 0, offset);
}
decompressed = out.toString();
} catch (IOException e) {
log.error("",e);
} finally {
if (ginzip != null) {
try {
ginzip.close();
} catch (IOException e) {
}
}
if (in != null) {
try {
in.close();
} catch (IOException e) {
}
}
if (out != null) {
try {
out.close();
} catch (IOException e) {
}
}
}
return decompressed;
}
}到此這篇關(guān)于springboot對(duì)壓縮請(qǐng)求的處理的文章就介紹到這了,更多相關(guān)springboot壓縮請(qǐng)求內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java swing實(shí)現(xiàn)電影購票系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了java swing實(shí)現(xiàn)電影購票系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01
gateway網(wǎng)關(guān)接口請(qǐng)求的校驗(yàn)方式
這篇文章主要介紹了gateway網(wǎng)關(guān)接口請(qǐng)求的校驗(yàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
Spring啟動(dòng)流程refresh()源碼深入解析
這篇文章主要給大家介紹了關(guān)于Spring啟動(dòng)流程refresh()源碼深入解析的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09
Java設(shè)計(jì)模式編程中的責(zé)任鏈模式使用示例
這篇文章主要介紹了Java設(shè)計(jì)模式編程中的責(zé)任鏈模式使用示例,責(zé)任鏈模式可以避免很多請(qǐng)求的發(fā)送者和接收者之間的耦合關(guān)系,需要的朋友可以參考下2016-05-05
java實(shí)現(xiàn)從方法返回多個(gè)值功能示例
這篇文章主要介紹了java實(shí)現(xiàn)從方法返回多個(gè)值功能,結(jié)合實(shí)例形式分析了集合類、封裝對(duì)象、引用傳遞三種實(shí)現(xiàn)方法,需要的朋友可以參考下2017-10-10
Spring?Cloud?Eureka基礎(chǔ)應(yīng)用及原理
這篇文章主要介紹了Spring?Cloud?Eureka基礎(chǔ)應(yīng)用,Eureka?Client中內(nèi)置一個(gè)負(fù)載均衡器,用來進(jìn)行基本的負(fù)載均衡,下面我們將通過搭建一個(gè)簡單的Eureka例子來了解Eureka的運(yùn)作原理,感興趣的朋友一起看看吧2022-05-05
解決eclipse中console控制選項(xiàng)不見了的方法
eclipse是一款用于編譯java語言的程序,利用這款軟件我們可以制作很多有趣的小程序,也可以制作一些大型的軟件項(xiàng)目,有的用戶在使用eclipse的時(shí)候會(huì)遇到console消失的情況,所以本文給大家介紹了解決eclipse中console控制選項(xiàng)不見了的方法,需要的朋友可以參考下2024-03-03

