基于快速开发,需求不稳定的情况, 我决定使用MongoDB作为存储数据库,搭配使用spring-data
因为快速开发,使用spring data可以直接在类上建表等其他操作,而且对于复合数据模型,MongoDB可以直接存储
gitee
github
<dependencies>
<dependency>
<groupId>joda-timegroupId>
<artifactId>joda-timeartifactId>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-data-mongodbartifactId>
dependency>
<dependency>
<groupId>cn.hutoolgroupId>
<artifactId>hutool-allartifactId>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starterartifactId>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-aopartifactId>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-testartifactId>
<scope>testscope>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
dependency>
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
dependency>
<dependency>
<groupId>com.alibabagroupId>
<artifactId>fastjsonartifactId>
dependency>
<dependency>
<groupId>com.google.guavagroupId>
<artifactId>guavaartifactId>
dependency>
dependencies>
基于maven dependencyManagement 版本控制如下:
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-dependenciesartifactId>
<version>2.1.2.RELEASEversion>
<type>pomtype>
<scope>importscope>
dependency>
<dependency>
<groupId>mysqlgroupId>
<artifactId>mysql-connector-javaartifactId>
<version>5.1.48version>
dependency>
<dependency>
<groupId>cn.hutoolgroupId>
<artifactId>hutool-allartifactId>
<version>4.5.16version>
dependency>
<dependency>
<groupId>org.mybatis.spring.bootgroupId>
<artifactId>mybatis-spring-boot-starterartifactId>
<version>2.0.0version>
dependency>
<dependency>
<groupId>com.alibabagroupId>
<artifactId>fastjsonartifactId>
<version>1.2.56version>
dependency>
<dependency>
<groupId>com.alibabagroupId>
<artifactId>druid-spring-boot-starterartifactId>
<version>1.1.9version>
dependency>
<dependency>
<groupId>com.google.guavagroupId>
<artifactId>guavaartifactId>
<version>19.0version>
dependency>
dependencies>
dependencyManagement>
docker run --restart="always" \
-d \
--name mongo\
-p 27017:27017\
-v /docker/mongo/data/db:/data/db\
mongo:latest --storageEngine wiredTiger
@Document(collection=“female”)
设置id: @Id
设置属性
构建索引
getter/setter
Female.java
@Document(collection = "female")
@Data
public class Female {
/**
* 主键
*/
@Id
private String id;
/**
* 姓名
*/
private String name;
/**
* 年龄
*/
private String age;
/**
* 哪种类型的女人;FemaleTypeEnums
*/
private Integer type;
/**
* 舔狗
*/
private List dogs;
/**
* 男朋友们
*/
@Indexed
private List boyFriends;
/**
* 男神们
*/
@Indexed
private List dreamers;
/**
* 创建时间
*/
@Indexed
private Date createTime;
/**
* 修改时间
*/
private Date modifiedTime;
}
创建Repository 继承于MongoRepository
根据规则来编写接口方法, spring data mongodb的dao 方法规则详细查看点这里,理论上用idea会提示出来的.
编写单元测试方法
创建Repository 继承于MongoRepository,编写接口方法
FemaleRepository.java
public interface FemaleRepository extends MongoRepository<Female,String> {
Page<Female> findAllByCreateTimeBetweenAndNameContaining(Date createTime, Date createTime2, String name, Pageable pageable);
Page<Female> findAllByCreateTimeBefore(Date createTime, Pageable pageable);
Page<Female> findAllByCreateTimeAfter(Date createTime, Pageable pageable);
Page<Female> findAllByCreateTimeBetween(Date start, Date end, PageRequest pageRequest);
}
单元测试方法
FemaleRepositoryTest.java
/**
* description: 添加测试数据
* author: suwenguang
* date: 2019-09-01
*/
@Test
public void addTestData() {
for (int i = 0; i < 1000; i++) {
Female entity = new Female();
entity.setName(RandomUtil.randomString(12));
LocalDate now = LocalDate.now();
LocalDate localDate = now.minusDays(RandomUtil.randomInt(4));
entity.setCreateTime(localDate.toDate());
femaleRepository.save(entity);
}
}
/**
* description: 测试查询构造器
* author: suwenguang
* date: 2019-09-01
*/
public void matching(){
//精确匹配和模糊匹配
Female probe = new Female();
ExampleMatcher matching = ExampleMatcher.matching()
.withMatcher("name", ExampleMatcher.GenericPropertyMatcher.of(ExampleMatcher.StringMatcher.CONTAINING))//模糊匹配
.withIgnorePaths("id")//忽略匹配id
;
PageRequest of = PageRequest.of(0, 10);
Page<Female> all = femaleRepository.findAll(Example.of(probe, matching), of);
System.out.println(JSON.toJSONString(all));
}
/**
* description: 测试范围查询
* author: suwenguang
* date: 2019-09-01
*/
@Test
public void findAllByCreateTimeAfter() {
LocalDate yesteday = new LocalDate().minusDays(3);
PageRequest of = PageRequest.of(0, 10);
List<Female> byCreateTimeAfter = femaleRepository.findAllByCreateTimeAfter(yesteday.toDate(), of);
System.out.println(JSON.toJSONString(byCreateTimeAfter));
}
/**
* description: 测试范围查询
* author: suwenguang
* date: 2019-09-01
*/
@Test
public void findByCreateTimeBetween() {
LocalDate localDate = new LocalDate();
Page<Female> byCreateTimeBetween = femaleRepository.findByCreateTimeBetween(localDate.minusDays(2).toDate(), localDate.toDate(), PageRequest.of(0, 10));
System.out.println(JSON.toJSONString(byCreateTimeBetween.getContent()));
}
(基于单表的复杂查询,多表复杂查询暂时不纳入讨论范围)
如果按照以上的用法,动态扩展多条件查询仍然不能够完美支持,会导致代码冗余,当然你如果使用mongoTemlate进行自己封装,另当别论.
那么为了实现动态扩展多条件查询,我去查看对应版本的官方文档,跳转点这里,看到可以集成querydsl作为扩展.
整合querydsl
使用dsl
1.Querydsl官网
2.querydsl集成文档
pom.xml配置引入依赖
<dependency>
<groupId>com.querydslgroupId>
<artifactId>querydsl-aptartifactId>
<version>${querydsl.version}version>
<scope>providedscope>
dependency>
<dependency>
<groupId>com.querydslgroupId>
<artifactId>querydsl-jpaartifactId>
<version>${querydsl.version}version>
dependency>
为什么要注释掉slf4j?
因为我的springboot项目已经引入了slf4j,没必要重复声明,自己可以通过idea的maven dependence查看是否有引入,没有则需要重新引入
在dao
的repository
中继承QuerydslPredicateExecutor
public interface FemaleRepository extends MongoRepository<Female,String>, QuerydslPredicateExecutor<Female> {
}
编写单元测试FemaleRepositoryTest.java
/**
* description: 多条件
* author: suwenguang
* date: 2019-09-01
*/
@Test
public void querydsl() {
PageRequest of = PageRequest.of(0, 10);
QFemale female = QFemale.female;
BooleanExpression createTimeBetween = female.createTime.between(LocalDate.now().minusDays(2).toDate(), LocalDate.now().minusDays(1).toDate());
BooleanBuilder builder = new BooleanBuilder(createTimeBetween);
BooleanExpression contains = female.name.contains("3");
builder.and(contains);
Page<Female> all = femaleRepository.findAll(builder,of);
System.out.println(all.getTotalElements());
System.out.println(JSON.toJSONString(all.getContent()));
}
如上所示, 这样子可以动态构造所需要的条件,多个范围查询也可以支持了!!!那么对于后台的搜索数据只需要一个接口就可以了
至于怎么实现,后面再继续整合 X-admin 2.2这个后端模板, 另外出一篇文章吧.
如果对上诉代码有问题或者有其他的扩展性问题,欢迎留下你的评论.