问题描述:我们都知道对于涉及钱的数据必须使用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);
}
}