MyBatis高效开发实践:自动填充公共字段的策略与实现
自动填充公共字段:
提升MyBatis应用的数据一致性和开发效率在企业级应用开发中,数据的一致性和完整性是数据库设计的重要考虑因素。公共字段,如创建人(createdBy)、创建时间(createdTime)、修改人(updatedBy)、修改时间(updatedTime)以及版本号(version),在实体类中的应用非常广泛。它们对于追踪记录的历史变化、实现乐观锁控制等方面至关重要。手动更新这些字段不仅效率低下,还容易出错。因此,实现自动填充这些公共字段变得尤为重要。
为什么要自动填充公共字段?
自动填充公共字段的好处包括:
- 提高开发效率:开发者无需在每次插入或更新数据时手动设置这些字段,减少了重复代码。
- 保证数据一致性:自动化处理可以减少人为错误,确保所有数据操作都遵循相同的规则。
- 易于维护:将公共字段的处理逻辑集中在一处,便于后期维护和修改。
- 审计追踪:自动记录数据的创建和修改信息,对于后续的数据审计和追踪非常有用。
如何在MyBatis中实现自动填充?
在MyBatis中,可以通过AOP(面向切面编程)来实现自动填充公共字段的功能。AOP允许定义切面(Aspect),在不改变现有业务逻辑的情况下,为程序动态地添加额外的行为。以下是如何使用Spring AOP在MyBatis中自动填充公共字段的步骤:
第一步:定义公共字段的基类
首先,定义一个包含公共字段的基类BaseEntity
:
import java.util.Date;
public class BaseEntity {
private String createdBy;
private Date createdTime;
private String updatedBy;
private Date updatedTime;
private Integer version;
// Getters and Setters
// ...
}
所有需要自动填充公共字段的实体类都应该继承自这个基类。
第二步:创建AOP切面
创建一个AOP切面FillPublicFieldsAspect
,并定义两个前置通知(Before advice)来在执行数据库操作前自动设置这些字段:
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.security.core.userdetails.UserDetails;
import org.springframework.stereotype.Component;
import java.util.Date;
@Aspect
@Component
public class FillPublicFieldsAspect {
// 假设有一个UserService用于获取当前登录的用户信息
@Autowired
private UserService userService;
@Before("execution(* com.derek.mall.mapper.*.insert*(..)) || execution(* com.derek.mall.mapper.*.add*(..))")
public void fillCreateFields(JoinPoint joinPoint) {
Object target = joinPoint.getTarget();
if (target instanceof MapperMethod) {
MapperMethod mapperMethod = (MapperMethod) target;
Object parameter = mapperMethod.getParameter();
if (parameter instanceof BaseEntity) {
BaseEntity entity = (BaseEntity) parameter;
String username = getCurrentUsername();
Date now = new Date();
if (entity.getCreatedBy() == null) {
entity.setCreatedBy(username);
}
if (entity.getCreatedTime() == null) {
entity.setCreatedTime(now);
}
entity.setVersion(1); // 初始版本号
}
}
}
@Before("execution(* com.derek.mall.mapper.*.update*(..)) || execution(* com.derek.mall.mapper.*.edit*(..))")
public void fillUpdateFields(JoinPoint joinPoint) {
Object target = joinPoint.getTarget();
if (target instanceof MapperMethod) {
MapperMethod mapperMethod = (MapperMethod) target;
Object parameter = mapperMethod.getParameter();
if (parameter instanceof BaseEntity) {
BaseEntity entity = (BaseEntity) parameter;
String username = getCurrentUsername();
entity.setUpdatedBy(username);
entity.setUpdatedTime(new Date());
// 版本号自增逻
// entity.setVersion(entity.getVersion() + 1);
}
}
}
private String getCurrentUsername() {
Object principal = SecurityContextHolder.getContext().getAuthentication().getPrincipal();
if (principal instanceof UserDetails) {
return ((UserDetails) principal).getUsername();
} else {
return principal.toString(); }
}
}
在fillCreateFields
和fillUpdateFields
方法中,可以使用Spring Security或其他机制来获取当前用户信息,并设置相应的字段。这个示例代码假设插入和更新操作的Mapper方法分别以insert
和update
为前缀。如果Mapper方法的命名不符合这个约定,需要调整@Before
注解中的切点表达式以匹配方法名
时序图:
在这个时序图中,展示了以下步骤:
- 用户通过控制器发送一个请求来执行增加或更新操作。
- 控制器将请求转发给服务层。
- 服务层在调用Mapper方法之前,由
FillPublicFieldsAspect
切面拦截。 FillPublicFieldsAspect
判断操作类型,如果是增加操作,则获取当前用户信息并填充createdBy
、createdTime
和version
字段;如果是更新操作,则填充updatedBy
和updatedTime
字段。- 填充完毕后,切面将控制权交还给Mapper。
- Mapper执行相应的SQL操作并将结果返回给服务层。
- 服务层将结果返回给控制器。
- 控制器将操作结果展示给用户。
第三步:集成Spring Security(可选)
如果使用Spring Security,可以通过下面的方法获取当前登录用户的用户名:
private String getCurrentUsername() {
Object principal = SecurityContextHolder.getContext().getAuthentication().getPrincipal();
if (principal instanceof UserDetails) {
return ((UserDetails) principal).getUsername();
} else {
return principal.toString();
}
}
第四步:在MyBatis Mapper中使用
确保Mapper方法使用了继承自BaseEntity
的实体类。当这些方法被调用时,切面将自动触发并填充公共字段。
public class User extends BaseEntity {
private String name;
private String address;
// Getters and Setters
// ...
}
结论
自动填充公共字段是提升数据一致性和开发效率的有效手段。通过使用Spring AOP和MyBatis,可以轻松实现这一功能,而无需在业务代码中手动设置这些字段。这种方法不仅减少了代码冗余,还提高了代码的可维护性和可靠性。如果MyBatis版本不支持直接获取MapperMethod
,可能需要调整代码以适应MyBatis版本。这个示例代码只是为了展示如何实现自动填充公共字段的基本思路,可能需要根据具体应用情况进行调整。
转载自:https://juejin.cn/post/7391745413823021075