问题描述:我们都知道对于涉及钱的数据必须使用BigDecimal类型进行存储,今天在查询mongo时仍然有精度问题,虽然我在代码中使用了Big Decimal类型,但mongo中使用的是double类型。我初步推断是mongoTemplate在类型转换时出现了问题,根源还是因为mongodb中使用了double类型来存储。但是我决定不了mongodb于是我只能从代码入手进行四舍五入的处理。
解决办法:自定义一个customer-convert从而在mongoTemplate在mapping的时候进行转换,从而不需要在各个地方进行修改。这里还有一个问题就是原来使用了springboot,于是也就没了mongo.xml的配置。自己实现的convert就没有地方注册了。最后我通过Java配置的方式自己new一个mongoTemplate并且注入到Ioc容器中,这样在我new mongoTemplate的时候就有机会注册convert了。
下面是具体代码
@Configuration public class MongodbConfiguration { @Value("${spring.data.mongodb.uri}") private String mongoUrl; @Bean public MongoMappingContext mongoMappingContext() { MongoMappingContext mappingContext = new MongoMappingContext(); return mappingContext; } @Bean @Primary public MongoDbFactory dbFactory1() throws UnknownHostException { return new SimpleMongoDbFactory(new MongoClientURI(mongoUrl)); } @Bean public MappingMongoConverter mappingMongoConverter1() throws Exception { DefaultDbRefResolver dbRefResolver = new DefaultDbRefResolver(this.dbFactory1()); MappingMongoConverter converter = new MappingMongoConverter(dbRefResolver, this.mongoMappingContext()); List<Converter> converters = new ArrayList<>(); converters.add(new DoubleToBigDecimalConverter()); CustomConversions customConversions = new CustomConversions(converters); converter.setCustomConversions(customConversions); return converter; } @Bean @Primary public MongoTemplate mongoTemplate() throws Exception { return new MongoTemplate(this.dbFactory1(), this.mappingMongoConverter1()); } } @ReadingConverter class DoubleToBigDecimalConverter implements Converter<Double,BigDecimal> { //在这里我们可以统一将double类型进行四舍五入并指定小数点后两位 @Override public BigDecimal convert(Double source) { BigDecimal bigDecimal = BigDecimal.valueOf(source); return bigDecimal.setScale(2,BigDecimal.ROUND_HALF_UP); } }