MyBatis高效开发实践:自动填充公共字段的策略与实现
2024-07-17 14:50 阅读(467)

自动填充公共字段:

提升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版本。这个示例代码只是为了展示如何实现自动填充公共字段的基本思路,可能需要根据具体应用情况进行调整。