Spring boot配置MongoDB以及Morphia踩坑记录

pom

因为项目中采用Morphia(MongoDBODM框架,对象-文档映射(object-document mapper)),因此需要在pom文件中引入相应依赖:

        <dependency>
            <groupId>dev.morphia.morphia</groupId>
            <artifactId>core</artifactId>
            <version>1.5.3</version>
        </dependency>

因为Morphia依赖于mongo-java-driver,因此无需在pom文件中继续导入mongo-java-driver.

配置

(1) 配置文件

application.yaml中配置如下所示:

  data:
    mongodb:
      database: {数据库名称}
      uri: mongodb://{用户名}:{密码}@{服务器地址}:27017/{数据库名称}

上述配置中,{}包围的地方请按照各自项目实际情况填写.

(2) 代码配置

@Data
@Configuration
public class MorphiaConfig {

    private MongoClient mongoClient;

    /**
     * 设置连接最大空闲时间(到达时间,连接关闭)
     * @return mongo client属性
     */
    @Bean
    public MongoClientOptions mongoClientOptions() {
        return MongoClientOptions.builder()
                .maxConnectionIdleTime(6000 * 5)
                .maxConnectionLifeTime(0)
                .build();
    }

    @Autowired
    public MorphiaConfig(MongoClient mongoClient) {
        this.mongoClient = mongoClient;
    }

    @Bean
    public Datastore datastore(@Autowired MongoClient mongoClient) {
        Morphia res = new Morphia();
        // 确定Mongo实体类的存放包名
        res.mapPackage("com.test.log.entity");
        Datastore datastore = res.createDatastore(mongoClient, "ops");
        //  为实体类创建索引
        datastore.ensureIndexes();
        return datastore;
    }
}

代码中无需自行配置MongoClient(Spring boot会自动完成MongoClient的初始化),当然也允许自行设置Mongo的连接参数.

实体类

项目中需要存放日志,并且通过MongoDBTTL特性设置过期时间,以便通过MongoDB完成日志删除任务.

@Builder
@NoArgsConstructor
@AllArgsConstructor
@Data
// 必须添加,以便Morphia识别此类为`MongoDB`的实体类,`logInfo`是表名
@Entity(value = "logInfo", noClassnameStored = true)
@Indexes({
        // expireTime存放日志过期时间,创建TTL index
        @Index(fields = @Field(value = "expireTime"), options = @IndexOptions(expireAfterSeconds = 0)),
        // 创建复合索引
        @Index(fields = {@Field("deviceId"), @Field("module"), @Field(value = "time", type = DESC)}),
        @Index(fields = {@Field("deviceId"), @Field(value = "time", type = DESC)})
})
public class LogInfo {
    @Id
    private ObjectId id;
    private Date time;
    @JsonIgnore
    private Date expireTime;
    private String module;
    private String level;
    private String deviceId;
    private String msg;
}

日志的存放和读取是在不同的项目完成,在查询MongoDB时遇到反序列化问题,异常如下所示:

WARN  dev.morphia.mapping.DefaultCreator - Class not found defined in dbObj:

查询源码分析,是因为Morphia在存储数据的时候,会将实体类名称存入数据库中.
查询时Morphia根据类名查找相应的实体类并进行反序列化.
因为不同的项目中,实体类的包名不一致导致出现以上错误.
解决方案比较简单,通过注解告知Morphia存储数据时不要存储包名即可,具体如下所示:

`@Entity(value = "logInfo", noClassnameStored = true)`.

CRUD操作

(1) 存储操作

    LogInfo logInfo = LogInfo.builder()
            .time(dateTime)
            .expireTime(dateExpireTime)
            .level(level)
            .module(module)
            .deviceId(deviceId)
            .msg(msg)
            .build();
            try {
        //插入日志到mongoDB
        datastore.save(logInfo);
    } catch (RuntimeException e) {
        log.error(AppStatus.IDB_WRITE_FAIL.getError(), e);
    }

(2) 查询操作

    final Query<LogInfo> query = datastore.createQuery(LogInfo.class).filter("deviceId = ",     request.getDeviceId());
    // 列表查询
    if (!CollectionUtils.isEmpty(request.getModules())) {
        query.filter("module in ", request.getModules().toArray());
    }
    // 根据时间查询,使用filter就不太恰当了
    if (null != request.getStartTime()) {
        query.field("time").greaterThanOrEq(request.getStartTime());
    }
    if (null != request.getFinishTime()) {
        query.field("time").lessThanOrEq(request.getFinishTime());
    }

    // 查询记录总数
    int count = query.count());

    // 设置排序规则,查询具体数据
    query.order(Sort.descending("time"));
    // 获取数据游标
    MongoCursor<LogInfo> logInfoMongoCursor = query.find(
            new FindOptions()
                .skip((request.getPageNum() - 1) * request.getPageSize())
                .limit(request.getPageSize())
        );
    // 通过数据游标,逐个获取数据记录
    while (logInfoMongoCursor.hasNext()) {
        logInfoPage.add(logInfoMongoCursor.next());
    }

删除以及更新操作就不再赘述,Morphia接口比较明晰,容易入门.

PS:
如果您觉得我的文章对您有帮助,可以扫码领取下红包或扫码支持(随意多少,一分钱都是爱),谢谢!

支付宝红包支付宝微信
764719-20190624102014468-193165117.png764719-20190624102025867-1690450943.png764719-20190624101626318-627523959.png

转载于:https://www.cnblogs.com/jason1990/p/11065800.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要在Spring Boot中使用MongoDB,需要执行以下步骤: 1. 添加MongoDB依赖项 在Maven项目中,可以在pom.xml文件中添加以下依赖项: ``` <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-mongodb</artifactId> </dependency> ``` 2. 配置MongoDB连接信息 在application.properties文件中添加MongoDB连接信息,例如: ``` spring.data.mongodb.uri=mongodb://localhost:27017/mydatabase ``` 其中,`mydatabase`是数据库的名称。 3. 创建MongoDB实体类 创建一个Java类,用于表示MongoDB中的一个文档。可以使用`@Document`注解来指定MongoDB中的集合名称,例如: ``` @Document(collection = "users") public class User { @Id private String id; private String name; private int age; // getter和setter方法 } ``` 在这个例子中,`User`类表示MongoDB中的`users`集合,包含`id`、`name`和`age`三个字段。 4. 创建MongoDB数据访问对象 创建一个接口,用于定义MongoDB的数据访问操作。可以使用Spring Data MongoDB提供的一些方法来实现常见的CRUD操作。例如: ``` @Repository public interface UserRepository extends MongoRepository<User, String> { List<User> findByName(String name); } ``` 在这个例子中,`UserRepository`接口继承`MongoRepository`,并且定义了一个`findByName`方法,用于按照`name`字段查询`User`对象。 5. 使用MongoDB数据访问对象 在需要使用MongoDB的地方,可以注入`UserRepository`对象,并且使用其提供的方法来进行数据访问。例如: ``` @Service public class UserService { @Autowired private UserRepository userRepository; public void saveUser(User user) { userRepository.save(user); } public List<User> findUsersByName(String name) { return userRepository.findByName(name); } } ``` 在这个例子中,`UserService`类注入了`UserRepository`对象,并且使用其提供的`save`和`findByName`方法来进行数据访问。 以上就是在Spring Boot配置MongoDB的步骤。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值