事件:
前端时不时反馈后端返回的中文数据出现了乱码,而且诡异的是乱码文字和位置都是随机出现(也就是说不一定可复现)!害怕😰!!!
分析原因:
1、起初以为只是简单的字符编码问题。随即添加了编码转换并且在yml配置中设置了以UTF-8格式返回。本以为这样就 so easy了!但是,如果仅仅这么简单,那这篇文章就没必要存在了~
2、之后发现事情并没有想象的那么简单,因为我把微服务本身到Gateway的字符编码都设置了一遍,前面的问题依旧存在!!!!!!!!!!!!!
3、最后测试微服务直接返回,发现没问题。随即将注意力转移至Gateway上,发现返回小数据量时不会有乱码问题,只有数据量大才会出现乱码问题!那么是不是因为随着项目的数据体量逐渐递增,而Gateway又是FluxWeb形式响应存在滑动窗口截取数据的问题呢?这样是不是就造成数据滑块内的中文被“拦腰截断”呢(utf-8一个中文3个字节)。
解决:
1、索性同时解决post | put请求body不能重复读的问题
2、解决乱码问题的其实就是这段:
.buffer().map(dataBuffer -> {
//DefaultDataBufferFactory join 乱码的问题解决
DataBufferFactory dataBufferFactory = new DefaultDataBufferFactory();
DataBuffer join = dataBufferFactory.join(dataBuffer);
以下是完整代码
@Slf4j
@Component
@EnableConfigurationProperties({FilterDataMaskingProp.class})
public class ResponseFilter implements Ordered, GlobalFilter {
@Autowired
private FilterDataMaskingProp dataMaskingProp;
@Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
ServerHttpRequest serverHttpRequest = exchange.getRequest();
ServerHttpResponse originalResponse = exchange.getResponse();
DataBufferFactory bufferFactory = originalResponse.bufferFactory();
ServerHttpResponseDecorator decoratedResponse = new ServerHttpResponseDecorator(originalResponse) {
@Override
public Mono<Void> writeWith(Publisher<? extends DataBuffer> body) {
String contentType = getDelegate().getHeaders().getFirst(HttpHeaders.CONTENT_TYPE);
Boolean flag = MediaType.APPLICATION_JSON_VALUE.equals(contentType) || MediaType.APPLICATION_JSON_UTF8_VALUE.equals(contentType);
if (body instanceof Flux && flag) {
Flux<? extends DataBuffer> fluxBody = (Flux<? extends DataBuffer>) body;
return super.writeWith(fluxBody.buffer().map(dataBuffer -> {
//DefaultDataBufferFactory join 乱码的问题解决
DataBufferFactory dataBufferFactory = new DefaultDataBufferFactory();
DataBuffer join = dataBufferFactory.join(dataBuffer);
// probably should reuse buffers
byte[] content = new byte[join.readableByteCount()];
join.read(content);
//释放掉内存
DataBufferUtils.release(join);
String result = new String(content, StandardCharsets.UTF_8);
byte[] uppedContent = new String(result.getBytes(), StandardCharsets.UTF_8).getBytes(StandardCharsets.UTF_8);
return bufferFactory.wrap(uppedContent);
}));
}
return super.writeWith(body);
}
};
return chain.filter(exchange.mutate().response(decoratedResponse).build());
}
@Override
public int getOrder() {
return -50;
}
}