spring-data-mongo 去除 _class 字段“失败”

本文介绍如何通过配置MongoTemplate成功去除MongoDB文档中的_class字段。文章对比了两种配置方法:一种成功移除了_class字段;另一种则未能实现目标,且目前未知具体原因。
部署运行你感兴趣的模型镜像

去除 _class 字段成功

@Configuration
public class MongoConfiguration {

    @Bean
    public MongoDbFactory mongoDbFactory() throws Exception {
        // TODO 初始化 factory 工作
        MongoDbFactory object = ...;
        return object;
    }

    @Bean
    public MongoTemplate mongoTemplate() throws Exception {
        MappingMongoConverter mongoConverter = new MappingMongoConverter(new DefaultDbRefResolver(mongoDbFactory()), new MongoMappingContext());
        mongoConverter.setTypeMapper(new DefaultMongoTypeMapper(null)); // typeKey 设置为 null。默认的无参构造为 _class
        return new MongoTemplate(mongoDbFactory(), mongoConverter);
    }

}

去除 _class 字段失败

@Configuration
public class MongoConfiguration {

    @Bean
    public MongoDbFactory mongoDbFactory() throws Exception {
        // TODO 初始化 factory 工作
        MongoDbFactory object = ...;
        return object;
    }

    @Bean
    public MongoOperations mongoTemplate() throws Exception {
        MappingMongoConverter mongoConverter = new MappingMongoConverter(new DefaultDbRefResolver(mongoDbFactory()), new MongoMappingContext());
        mongoConverter.setTypeMapper(new DefaultMongoTypeMapper(null)); // typeKey 设置为 null。默认的无参构造为 _class
        return new MongoTemplate(mongoDbFactory(), mongoConverter);
    }

}

代码的区别

唯一的不同之处为 mongoTemplate() 方法。返回的类型范围不同,
返回向上转型后的类型 MongoOperations,该方式去除失败
返回原始类型 MongoTemplate,该方式成功

目前未发现该 bug 产生的原因。

您可能感兴趣的与本文相关的镜像

Stable-Diffusion-3.5

Stable-Diffusion-3.5

图片生成
Stable-Diffusion

Stable Diffusion 3.5 (SD 3.5) 是由 Stability AI 推出的新一代文本到图像生成模型,相比 3.0 版本,它提升了图像质量、运行速度和硬件效率

<think>我们正在使用Spring Boot和MongoDB的BulkOperations进行批量插入,但默认情况下,Spring Data MongoDB会为每个文档添加一个`_class`属性,用于存储文档的Java类型。但在某些场景下,我们可能不希望这个字段被存储到数据库中。 解决方案: 1. 我们可以通过配置映射上下文(MongoMappingContext)来禁用`_class`字段的存储。 2. 另一种方法是在实体类上使用注解`@Document(collection = "yourCollection")`,并通过`@TypeAlias`来指定类型别名,但这并不能直接去掉`_class`字段。要去掉,我们需要在映射上下文中设置默认类型映射为null。 但是,对于批量操作,我们使用BulkOperations,我们可以通过配置MongoTemplate来禁用`_class`字段的自动添加。 具体步骤: a. 配置MongoTemplate,设置DefaultMongoTypeMapper的typeKey为null,这样就不会存储`_class`字段。 b. 在批量插入时,使用这个配置好的MongoTemplate。 示例代码: 首先,配置MongoTemplate: ```java @Configuration public class MongoConfig extends AbstractMongoClientConfiguration { @Override public String getDatabaseName() { return "test"; } @Bean public MongoTemplate mongoTemplate() throws Exception { MappingMongoConverter converter = new MappingMongoConverter( new DefaultDbRefResolver(mongoDbFactory()), new MongoMappingContext()); // 这里设置_typeMapper为null,这样就不会有_class字段 converter.setTypeMapper(new DefaultMongoTypeMapper(null)); return new MongoTemplate(mongoDbFactory(), converter); } } ``` 然后,在批量插入时,使用这个MongoTemplate: ```java @Autowired private MongoTemplate mongoTemplate; public void bulkInsertWithoutClassField(List<YourEntity> entities) { BulkOperations bulkOps = mongoTemplate.bulkOps(BulkOperations.BulkMode.UNORDERED, YourEntity.class); for (YourEntity entity : entities) { InsertOneOperation<YourEntity> insertOne = new InsertOneOperation<>(entity); bulkOps.insert(Collections.singletonList(insertOne)); } bulkOps.execute(); } ``` 注意:上面的代码中,我们使用了`BulkOperations`的插入操作。由于我们在MongoTemplate中配置了类型映射器将类型键设置为null,所以插入的文档中不会包含`_class`字段。 另外,需要注意的是,禁用`_class`字段后,当我们从数据库读取文档时,Spring Data MongoDB将无法自动将文档映射到具体的Java类型(除非我们提供其他方式,比如使用@TypeAlias并配置类型映射器)。因此,如果我们的集合中存储了多种类型的文档,并且我们依赖`_class`字段进行多态反序列化,那么禁用这个字段可能会导致问题。如果我们的集合中只存储一种类型的文档,并且我们明确知道要映射到哪个类,那么我们可以禁用`_class`字段。 另一种方法是,在实体类上使用`@Document`注解,并设置`type`属性为空字符串,例如: ```java @Document(collection = "yourCollection", type = "") public class YourEntity { // ... } ``` 但是这种方法在批量插入时可能不会生效,因为Spring Data MongoDB的默认行为还是会添加`_class`字段,除非我们全局配置了类型映射器。 因此,推荐使用配置MongoTemplate的方式。 引用说明: 在Spring Data MongoDB中,我们可以通过配置`MappingMongoConverter`的`TypeMapper`来控制类型信息的存储。将`DefaultMongoTypeMapper`的类型键设置为null,即可禁用`_class`字段[^1]。 另外,关于批量操作,MongoDB提供了批量写入的能力,Spring Data MongoDB的`BulkOperations`是对此的封装[^3]。 因此,通过上述配置,我们可以在批量插入时避免新增`_class`属性。 相关问题: 1. 如何在Spring Data MongoDB中配置多数据源? 2. 批量操作时,如何提高性能? 3. 除了去掉`_class`字段,还有哪些常见的MongoDB文档映射配置?</think>### 避免 Spring Boot MongoDB 批量插入时生成 _class 属性 在 Spring Data MongoDB 中,默认会为每个文档添加 `_class` 属性(存储 Java 类名),用于反序列化时类型识别。若需在批量插入时禁用此字段,可通过以下方案实现: #### 方案 1:全局配置(推荐) 在 `MongoTemplate` 中配置类型映射器,禁用 `_class` 字段: ```java @Configuration public class MongoConfig { @Bean public MongoTemplate mongoTemplate(MongoDatabaseFactory factory) { MappingMongoConverter converter = new MappingMongoConverter( new DefaultDbRefResolver(factory), new MongoMappingContext() ); // 关键配置:禁用 _class 字段 converter.setTypeMapper(new DefaultMongoTypeMapper(null)); return new MongoTemplate(factory, converter); } } ``` #### 方案 2:实体级配置 在实体类使用 `@TypeAlias` + 全局类型映射器: ```java @Document @TypeAlias("user") // 定义类型别名 public class User { @Id private String id; private String name; } // 配置类中追加 converter.setTypeMapper(new DefaultMongoTypeMapper("_alias")); // 使用自定义字段名 ``` #### 批量插入实现 使用配置好的 `MongoTemplate` 执行批量操作: ```java public void bulkInsert(List<User> users) { BulkOperations bulkOps = mongoTemplate.bulkOps( BulkOperations.BulkMode.UNORDERED, User.class ); users.forEach(user -> { bulkOps.insert(user); // 自动应用无 _class 的映射规则 }); BulkWriteResult result = bulkOps.execute(); // 执行批量操作 } ``` #### 注意事项 1. **反序列化影响** 禁用 `_class` 后,查询结果需明确指定返回类型: ```java mongoTemplate.find(new Query(), User.class); // 必须显式声明实体类型 ``` 2. **多态类型限制** 若集合存储多种实体类型,需保留 `_class` 或使用 `@TypeAlias` 维护类型信息[^1]。 3. **性能优势** 批量操作相比单条插入可提升 10-50x 性能,尤其在大数据量场景[^3]。 > **提示**:若需临时保留 `_class` 字段(如特定集合),可通过 `@Document(collection = "users")` 单独配置,不影响全局设置。 --- ### 相关问题 1. 如何优化 Spring Data MongoDB 的批量写入性能? 2. 禁用 `_class` 字段后如何处理多态文档的反序列化? 3. 除了批量插入,MongoDB 还有哪些高效数据写入方式? 4. 如何为不同集合单独配置 `_class` 字段的生成规则? [^1]: Spring Data MongoDB 通过 `_class` 实现实体类型识别,禁用后需显式指定结果类型。 [^3]: MongoDB 批量操作显著减少网络开销,适合大数据写入场景。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值