1 MongoDB中的文档设计
以MongoDB做数据库进行CRUD操作,先要思考怎么进行MongoDB的文档设计
在进行文档设计之前,先回顾数据库的三范式
1.1 回顾数据库三范式
简单总结来说,数据库三范式如下:
- 在数据库中,每个表的一个列中不能存在多个值
- 每个表都必须要拥有一个且只能拥有一个唯一主键
- 一个表的外键只能关联其他表的主键
1.2 打破第三范式
现有如下需求:
员工表 : id , name , dept_id
部门表 : id , dept_name
要求 查询获取到员工信息与部门信息
按照之前MySQL的经验,表设计和SQL设计如下:
class Employee{
Long id;
String name;
Long dept_id;
}
class Department{
Long id;
String dept_name;
}
select e.*,d.* from employee e left join department d where e.dept_id = d.id
但是这样设计有一个缺点,**那就是当数据量非常庞大之后,其多表查询产生的 笛卡尔积 也会非常庞大 , 那应该如何简化呢? **
如果我们将员工表这么设计:
class Employee{
Long id;
String name;
Long dept_id;
String dept_name;
}
这样不就可以由多表查询–>单标查询,避免产生笛卡尔积 , 直接查询这个员工的部门信息了?
因为这样的设计违反了数据库的第三范式 , 因此我们将这个操作称之为 打破第三范式
其中, 添加的 dept_name 称之为 冗余字段
当然这个操作也存在缺点 : 添加与更改操作,需要维护冗余字段
因此,其适用场景为 : 当需要进行的 DQL 操作量远远大于 DML 操作时
MongoDB推荐使用冗余字段的方式,减少表之间的管理
1.3 MongoDB文档设计
1.3.1 一对一
现在有如下类:
husband : 丈夫
wife : 妻子
显然,丈夫和妻子之间是一对一的关系
MySQL中的表设计如下:
husband: | | | wife: | ||
id | name | | | id | name |
1 | Jack | | | 1 | Rose |
2 | Trump | | | 2 | Bitch |
两个表之间,采用相同的主键设计
相对应的类设计如下:
class husband{
Long id;
String name;
Wife wife;
}
class wife{
Long id;
String name;
}
在MongoDB中,对应一对一的方式,一般使用 内嵌的方式
husband:
{
id:xx ,
name:xx ,
wife:{
id:xx ,
name:xx
}
}
但是这种设计,当需要将 内嵌字段 拉出来进行独立需求时,会显得非常复杂,于是当遇到这种需求,也只能像MySQL一样了
1.3.2 一对多/多对一
现在有如下类:
employee : 员工
department : 部门
显然员工和部门之间是多对一的关系
MySQL中的表设计如下:
employee: | | | department: | |||
id | name | dept_id | | | id | name |
1 | Jack | 2 | | | 1 | Sell |
2 | Trump | 1 | | | 2 | Fire |
相对应的类设计如下:
class employee{
Long id;
String name;
Department dept;
}
class department{
Long id;
String name;
}
在MongoDB中,还是推荐使用内嵌的方式 :
Employee:
{
id:xx ,
name:xx ,
department:{
id:xx ,
name:xx
}
}
缺陷和一对一时是一样的
需要注意的是, 一对一关系和一对多关系,在 MongoDB 中,也都支持使用 冗余字段 的写法进行
1.3.3 多对多
现在有如下类 :
student : 学生
teacher : 老师
显然他们之间是 多对多 关系
在MySQL中,会采用中间表来关联他们的关系 :
student_id | teacher_id |
1 | 1 |
1 | 2 |
2 | 1 |
对应类设计如下 :
class student{
Long id;
String name;
List<Teacher> teachers; // 此处推荐使用集合而不是数组
}
class teacher{
...
}
在MongoDB中,会选择使用数组来保存这种 多对多 关系 (也可以用来保存 一对多 关系)
当teacher的数量不多时:
student{
id:xx,
name:xx,
teacher:[
{id,name},
{id,name},
...
]
}
但是,这种设计有一个缺点,当teacher的字段和数量很多时,这种书写不仅麻烦,而且有一个问题
我们知道在MongoDB中,一个文档的大小不能超过 16M , 当 teacher 数量庞大时,我们可以使用这种书写方式 :
student{
id:xx,
name:xx,
teacher:[
"teacherId1",
"teacherId2",
...
]
}
2 SpringBoot集成MongoDB
- 新建 Maven 项目
- pom.xml 中指定父依赖,并导入 MongoDB 自动配置依赖
- 新建 application.properties 配置文件并配置数据库连接
- 创建对应实体类并在实体类上贴上 @Document(" 集合名 ") 注解
- 创建 repository 包并在里面创建 实体类Repository 接口 , 使该接口继承 MongoRepository 接口,该接口泛型 第一个为操作的对象,第二个为操作对象主键类型
- 创建SpringBoot主配置类并启动SpringBoot
- 书写业务层接口及其实现类
3 继承后基本CURD操作及其原理
书写业务层接口及其实现类如下:
public interface IUserService {
void save(User user);
void update(User user);
void delete(String id);
User get(String id);
List<User> list();
}
@Service
public class UserServiceImpl implements IUserService {
@Autowired
private UserRepository repository;
@Override
public void save(User user) {
repository.save(user);
}
@Override
public void update(User user) {
repository.save(user);
}
@Override
public void delete(String id) {
repository.deleteById(id);
}
@Override
public User get(String id) {
Optional<User> op = repository.findById(id);
return op.orElse(null);
}
@Override
public List<User> list() {
return repository.findAll();
}
}
这里有几个问题和细节:
- save和update,调用的方法都是 repository.save() , 区别就在于参数中有没有带 id , 带了就是改 , 没带就是增
- 新增操作时,会发现数据库表格中新增了一列 : _class :
其原因是 : 插入数据时,会给所有 属性 增加列,因为其他属性都有对应列了,所以没有新增
但是我们知道,所有类都继承于 Object 类,该类有一个方法 getClass() , 根据属性的定义 , 于是 所有类都有一个隐藏的属性 class , 所以便新增了一个 _class 列
- 新增操作时,如果这么书写
@Test
public void testUpdate(){
User user = new User();
user.setId("5f2563b01e350000400009bb");
user.setName("Potter");
userService.update(user);
}
那么会产生这种结果:
两列数据消失
这是因为 spring-data-mongodb 里面的更新操作,称之为全量更新,一旦执行该操作,所有字段都会被更新
其实其底层就是拿新数据覆盖了原来的数据,所以一旦某些字段我们没有为其赋值,更新后他们就会变为 N/A
想要正常更新,就需要 先查询到对应数据,再将新数据set进去,再利用这个对象进行更新:
// 先获取想要更新的对象
User user = userService.get("5f2563b11e350000400009bd");
// 将我们想要更新的值set进去
user.setName("法海");
// 再更新
userService.update(user);
- 现在来到最重要的一个点 : 为什么 UserRepository 接口不书写任何方法,只是继承了 MongoRepository 接口, 业务层就可以直接调用 CRUD 方法?
查看 UserRepository 接口继承体系:
在上面那些接口中定义了 CRUD 方法,所以 UserRepository 也继承了对应 CRUD 方法
- 另外 , UserRepository 接口在表面上没有创建对象,也可以调用方法,其实是因为 spring 容器在底层通过 JDK 动态代理的方式,帮我们创建了实现类,通过
.getClass()
方法可以看出来:
SpringBoot会自动将继承了 MongoRepository 接口 的接口,进行动态代理 ,创建这个接口的代理实现对象
4 JPA查询规范
当我们想通过姓名等查询用户信息怎么办?
我们只需要在 UserRepository 接口中,创建对应的符合 JPA 查询规范的方法就可以
JPA查询规范:
定义符合JPA约束的查询方法,其格式如下:
前缀 + 属性名 + 操作符
其中,前缀需要以 findBy / queryBy 开头
属性名为 对应表的一个列名,并将首字母大写
操作符是类似于 and 和 or 等
其原理是,spring在动态代理时,根据JPA查询规范,解析我们定义的方法,拼接出对应的MQL语句
JPA规范如下 :
5 MongoTemplate
使用JPA查询,简单也方便,但是当查询条件增多导致JPA无法进行拼接时,此时就需要使用到 :
MongoTemplate 模板方式
这是 spring-data-mongodb 的第二种查询方式,他是 spring-data 规范下创建的子项目,其核心类为 MongoTemplate
其中,还有一些常用的其他类:
- Query ( org.springframework.data.mongodb.core.query ) : 表示查询对象
- Criteria : 用来封装查询条件,其静态方法和非静态方法都有相关查询条件的方法
- Pageable : 用来封装分页查询结果的类
使用 MongoTemplate 的主要流程为 :
- 注入 MongoTemplate 类
- 创建 查询对象类Query 的对象
- 根据需求决定是否创建 Criteria 类的对象 ,不创建就使用其静态方法封装查询条件,创建可以使用其非静态方法封装查询条件,非静态方法中主要涉及到 运算符 等方法
- 将查询条件封装到查询对象中 :
query.addCriteria(criteria)
- 使用 MongoTemplate 的对象调用 find() 查询方法,传入 查询对象,返回值类型,集合名(这个可以省略)
案例:
// 查询 name=法海 的用户
@Test
public void testQuery2(){
// 查询对象
Query query = new Query();
// 用来封装条件
Criteria criteria = Criteria.where("name").is("法海");
// 将条件封装到查询对象中
query.addCriteria(criteria);
// 执行DQL语句
// List<User> users = mt.find(query, User.class,"users");
// 第三个参数 users 实际上可以省略,因为对应实体类上贴了 @Document 注解指定了集合类型
List<User> users = mt.find(query, User.class);
System.err.println(users);
}
// 分页查询,显示第二页,每页显示4条信息,并按照id升序排序
@Test
public void testQuery3(){
Query query = new Query();
/*
// 指定排序规则 : 以 id 进行排序,升序排序, Direction是一个枚举类
Sort sort = Sort.by(Sort.Direction.ASC,"id");
// 将排序规则加到查询对象中
query.with(sort);
// 指定分页条件并加到查询对象中
query.skip(4);
query.limit(4);
*/
// 上面的操作可以替换如下:
/**
* 此时的参数
* 第一个: 相当于 currentPage
* 第二个: 相当于 pageSize
* 第三个: 排序规则
* 第四个: 排序对象
*/
Pageable pageable = PageRequest.of(1, 4, Sort.Direction.ASC, "id");
query.with(pageable);
List<User> users = mt.find(query, User.class);
users.forEach(System.err::println);
}
// 查询所有 name=法海 或者 age<28 的用户
@Test
public void testQuery4(){
Query query = new Query();
Criteria criteria = new Criteria();
// 此时就需要用到 or 或者运算符了
// 使用 or 运算符,在里面拼接查询条件
criteria.orOperator(
Criteria.where("name").is("法海"),
Criteria.where("age").lt(28)
);
// 设置完条件再放进来
query.addCriteria(criteria);
List<User> users = mt.find(query, User.class);
users.forEach(System.err::println);
}
// 查询 name中含有 da 并且 25<age<30 的用户
@Test
public void testQuery5(){
Query query = new Query();
Criteria criteria = new Criteria();
// 根据需求,此时用 and 运算符
criteria.andOperator(
Criteria.where("name").regex("da"),
Criteria.where("age").gt(25).lt(30)
);
query.addCriteria(criteria);
List<User> users = mt.find(query, User.class);
users.forEach(System.err::println);
}
t.find(query, User.class);
users.forEach(System.err::println);
}
// 查询 name中含有 da 并且 25<age<30 的用户
@Test
public void testQuery5(){
Query query = new Query();
Criteria criteria = new Criteria();
// 根据需求,此时用 and 运算符
criteria.andOperator(
Criteria.where(“name”).regex(“da”),
Criteria.where(“age”).gt(25).lt(30)
);
query.addCriteria(criteria);
List users = mt.find(query, User.class);
users.forEach(System.err::println);
}