springboot mongo 不插入_class

本文介绍了一个Spring Boot项目中自定义MongoDB配置的方法。通过创建SpringMongoConfig类并使用@Bean注解来实例化MongoTemplate,连接到指定的MongoDB服务器地址(如192.168.200.100:27017),并设置数据库名为yywdb。此外,还展示了如何通过SimpleMongoDbFactory和MappingMongoConverter进行数据库工厂和类型映射配置,以实现MongoDB的数据操作。

参考:https://www.2cto.com/database/201608/533434.html

//会忽略 springboot 配置文件里的数据库配置

@Configuration
public class SpringMongoConfig {
    public @Bean MongoTemplate mongoTemplate() throws Exception {
        //权限验证 MongoCredential
       // MongoCredential credential = MongoCredential.createCredential("root", "mydb", "root".toCharArray());
       // ServerAddress serverAddress = new ServerAddress("localhost", 27017);
        ServerAddress serverAddress = new ServerAddress("192.168.200.100", 27017);
        MongoClient mongoClient = new MongoClient(serverAddress);
        SimpleMongoDbFactory mongoDbFactory = new SimpleMongoDbFactory(mongoClient, "yywdb"); // yywdb 是数据库名
        DbRefResolver dbRefResolver = new DefaultDbRefResolver(mongoDbFactory);
        MappingMongoConverter converter = new MappingMongoConverter(dbRefResolver, new MongoMappingContext());
        //不插入_class
        converter.setTypeMapper(new DefaultMongoTypeMapper(null));
        MongoTemplate mongoTemplate = new MongoTemplate(mongoDbFactory, converter);
        return mongoTemplate;
    }
  
}

<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 批量操作显著减少网络开销,适合大数据写入场景。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值