为什么要关闭?
如果不关闭,框架会将数据放到Tomcat的线程里的threadLocal里。由于Tomcat的线程正常情况不会回收,所以这个占用会一直存在
如何关闭?
如果是自己定义的ObjectMapper,其实关闭还是比较容易的
private static ObjectMapper objMapper = new ObjectMapper();
static {
//虽然提示过期了,但是还是生效的
objMapper.getFactory().disable(JsonFactory.Feature.USE_THREAD_LOCAL_FOR_BUFFER_RECYCLING);
}
最最关键的上面的代码只是关闭了我们自己定义的ObjectMapper。别忘了,springboot本身的序列化也是由jackson完成的,如果你也想关掉这个参数,这个就需要下面这样操作。
@Configuration
public class CustomerJackson {
@Bean
@Primary
@ConditionalOnMissingBean(ObjectMapper.class)
public ObjectMapper objectMapper(Jackson2ObjectMapperBuilder builder) {
ObjectMapper objMapper = builder.createXmlMapper(false).build();
objMapper.getFactory().disable(JsonFactory.Feature.USE_THREAD_LOCAL_FOR_BUFFER_RECYCLING);
return objMapper;
}
}
如何验证是否关闭成功
通过debug的方式进行验证。
这2个地方都会用到springboot内置的序列化,也就是Jackson。将断点打在如下图所示位置
方法是com.fasterxml.jackson.core.JsonFactory#_getBufferRecycler
看看1709行的条件是真是假,如果不进入1710行就表示关闭成功了,如果进入了就表示关闭没有成功
然后再将dump文件分析一下,你会发现threadLocal内已经没有数据了。
小结
其实一开始我关掉了ObjectMapper后分析dump文件,发现SoftReference<BufferRecycler>还是存在。然后就怀疑是不是disable这个命令没起作用。现在看,其实是起作用了,之所以还会在dump文件看到是springboot内置的ObjectMapper放的。所以只有把所有的ObjectMapper里的这个参数关掉在threadLocal中就看不见SoftReference<BufferRecycler>了