[TOC]
# 1.`@Enumerated`
该注解可以让一个枚举类型变量作为数据库中的一个字段。
```java
package jakarta.persistence;
public @interface Enumerated {
//EnumType.ORDINAL 将枚举索引作为值插入到数据库中
//EnumType.STRING 将枚举名称作为值插入到数据库中
EnumType value() default EnumType.ORDINAL;
}
```
**1. 创建一个枚举类型**
```java
public enum GenderEnum {
男生,
女生
}
```
**2. 将该枚举类作为数据库中的一个字段**
```java
public class Student implements Serializable {
@Enumerated(value = EnumType.ORDINAL)
@Column(name = "gender")
private GenderEnum gender;
}
```
**3. 在插入数据时**
:-: ![](https://img.kancloud.cn/4f/5f/4f5fac96d71e72c997c50d7ee997fb03_1501x186.png)
EnumType.ORDINAL 插入的是枚举的索引
<br/>
:-: ![](https://img.kancloud.cn/8c/7d/8c7dbad1270d23e9afca2c5df7c5c747_1528x190.png)
EnumType.STRING 插入的是枚举名
<br/>
# 2. `@Fetch`
多表查询时使用注解`@Fetch`可以决定查询方式,如果不使用该注解,默认查询方式应该为`FetchMode.SELECT`。
```java
package org.hibernate.annotations
public @interface Fetch {
//FetchMode.SELECT 先查主表,在用主表ID去查从表,共查两次
//FetchMode.JOIN left join查询,只查一次
//FetchMode.SUBSELECT 子查询
FetchMode value();
}
```
```java
public class Student implements Serializable {
@Fetch(value = FetchMode.SELECT)
@OneToMany(mappedBy = "student")
private List<Score> scoreList;
}
```
<br/>
# 3.`@MappedSuperclass`
```
jakarta.persistence.MappedSuperclass 注解可以将一些公共字段抽取出来封装在一个父类中
```
<br/>
**1. 将公共字段抽取到一个抽象父类中**
```java
@Getter
@Setter
@MappedSuperclass
public abstract class BaseEntity implements Serializable {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Integer id;
@Column(name = "create_time")
private Date createTime;
@Column(name = "update_time")
private Date updateTime;
}
```
**2. 其他实体类继承抽象父类**
```java
public class Student extends BaseEntity {
@Column(name = "stu_name")
private String name;
@Column(name = "age")
private Integer age;
...
}
```
![](https://img.kancloud.cn/05/f9/05f9f305a3a714d62a5b6054cdf83638_1686x199.png)
<br/>
# 4. `@PrePersist/@PreUpdate`
```java
@Getter
@Setter
@MappedSuperclass
public abstract class BaseEntity implements Serializable {
...
@Column(name = "create_time")
private Date createTime;
@Column(name = "update_time")
private Date updateTime;
/**
* 在新增前对createTime、updateTime赋值
*/
@PrePersist
public void beforeAdd() {
System.out.println("[beforeAdd]");
SimpleDateFormat dFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date now = new Date();
try {
if (this.createTime == null) {
this.createTime = dFormat.parse(dFormat.format(now));
}
if (this.updateTime == null) {
this.updateTime = dFormat.parse(dFormat.format(now));
}
} catch (ParseException ex) {
}
}
/**
* 在更新前对updateTime赋值
*/
@PreUpdate
public void beforeUpdate() {
System.out.println("[beforeUpdate]");
if (this.updateTime != null) {
return;
}
SimpleDateFormat dFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
try {
this.updateTime = dFormat.parse(dFormat.format(new Date()));
} catch (ParseException ex) {
}
}
}
```
<br/>
# 5. `@DynamicUpdate/@DynamicInsert`
```
org.hibernate.annotations.DynamicInsert 生成动态的 insert 语句
org.hibernate.annotations.DynamicUpdate 生成动态的 update 语句
```
```java
...
@DynamicUpdate
@DynamicInsert
public class Student {
@Column(name = "stu_name")
private String name;
@Column(name = "age")
private Integer age;
...
}
```
这里只新增/更新 age 一个字段,sql 语句如下:
```sql
-- 不标记注解 @DynamicInsert
insert into tb_student (age,stu_name) values (?,?)
-- 标记注解 @DynamicInsert,
insert into tb_student (age) values (?)
-- 不标记注解 @DynamicUpdate
update tb_student set age=?,stu_name=? where id=?
-- 标记注解 @DynamicUpdate
update tb_student set age=? where id=?
```
<br/>
>[info]放在父类中无效。
```java
@Getter
@Setter
@DynamicUpdate
@DynamicInsert
@MappedSuperclass
public abstract class BaseEntity implements Serializable {
...
}
public class Student extends BaseEntity {
@Column(name = "stu_name")
private String name;
}
```
- Spring
- Spring是什么
- Spring与EJB对比
- Spring的组成
- 首个Spring程序
- IoC控制反转
- 什么是IoC
- IoC编程
- 依赖注入方式
- 不同变量注入
- AOP面向切面编程
- AOP思想
- AOP实现原理
- AOP关键术语
- AOP编程
- 5种增强方式
- 切入点规则
- 自动装配
- Spring注解开发
- Bean注解
- AOP注解
- 完全注解
- 配置文件拆分
- SpringBean
- Bean常用属性
- Bean作用域
- Bean生命周期
- SpringBoot
- SpringBoot是什么
- 项目创建
- 配置文件
- 配置类型
- 读取配置
- 占位符
- 多环境配置
- 配置优先级
- 更改配置文件
- 自定义IoC容器
- 常用组件
- ApplicationContextAware
- CommandLineRunner
- Boot[Web]
- 引入模板引擎
- 静态资源访问
- 指定首页
- JSP支持
- 注册拦截器
- 注册Servlet组件
- 注册Servlet
- 注册过滤器
- 注册监听器
- 拦截器与过滤器区别
- 文件上传
- 文件下载
- 变更服务器
- Controller层封装
- HttpServletRequest
- 获取请求行
- 获取请求头
- 获取请求体
- Boot[自动配置]
- 自动配置是什么
- 自动配置报告
- 关闭自动配置
- 条件注解
- Boot[场景启动器]
- 场景启动器是什么
- 自定义场景启动器
- Boot[日志]
- 日志框架
- 日志级别
- 日志配置
- 配置文件
- 切换日志
- Boot[邮件任务]
- Boot[定时任务]
- cron表达式
- 起步
- 任务并行
- 注解Scheduled参数
- Boot[异步任务]
- 起步
- 注意事项与原理
- 自定义线程池
- Boot[缓存]
- JSR107缓存技术
- Spring缓存抽象
- 缓存注解
- SpEL表达式
- 起步
- 自定义key生成器
- 工作原理
- Boot[Redis]
- 起步
- 序列化机制
- Boot[Jdbc]
- 起步
- 两个模板类
- JdbcTemplate
- 增删改
- 查询
- NamedParameterJdbcTemplate
- 增删改
- 查询
- 自定义JdbcTemplate
- Boot[JPA]
- SpringDataJPA是什么
- 与JPA、Hibernate的关系
- 起步
- SpringDataJPA原理
- 查询方式
- 方法命名规则查询
- 限制查询结果查询
- 注解Query查询
- 命名参数查询
- SpEL表达式查询
- 原生查询
- 更新与删除
- 查询指定字段
- Specification动态查询
- 分页查询与排序
- 多表查询
- 一对一查询
- 一对多查询
- 多对多查询
- Specification查询
- Query注解查询
- 主键策略
- 单独主键
- 联合主键
- 级联操作
- 加载规则
- 审计功能
- 常用注解
- 避坑指南
- Boot[JSR303]
- JSR303是什么
- 常用约束
- 起步
- 简单校验
- 嵌套校验
- 分组校验
- 自定义约束注解
- 自定义校验工具
- Spring事务
- 事务的作用
- 起步
- 事务参数
- SpringDoc文档
- SpringDoc是什么
- 起步
- 自定义配置
- 常用Doc注解
- JSR303文档
- knife4j文档
- 常用配置
- Boot[RabbitMQ]
- 起步
- Fanout交换机类型
- Direct交换机类型
- Topic交换机类型
- 延迟队列插件
- RabbitListener监听方法
- JWT认证
- 认证流程
- 起步
- 密码加密
- JWT认证实现