借助网上参考的内容,写出自己记录操作日志的心得!!
我用的是ssm项目使用aop记录日志;这里用到了aop的切点 和 自定义注解方式;
1、建好数据表:
数据库记录的字段有: 日志id 、操作人、操作人IP、操作时间、操作方法、操作哪个控制层或者服务层、操作说明(记录用户操作的详情说明)、类型、异常信息
2、在spring重配置如下:
因为在spring里我配置了记录服务层的切点; 所以我在spring-mvc 里面 重新配置了一个可以扫描控制层的切点,使用的是cglib 代理:
3、接下来写自定义的注解
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
@Documented
public@interface ArchivesLog {
/**
* 操作说明
*/
public String operteContent() default “”;
}
4、拿到切点的内容,并且在此存库
packagecom.zhkj.jtpdms.log;
importjava.lang.reflect.Method;
importjavax.servlet.http.HttpServletRequest;
importjavax.servlet.http.HttpSession;
importorg.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.*; importorg.slf4j.Logger; importorg.slf4j.LoggerFactory; importorg.springframework.beans.factory.annotation.Autowired; importorg.springframework.stereotype.Component; importorg.springframework.web.context.request.RequestContextHolder; importorg.springframework.web.context.request.ServletRequestAttributes; importcom.zhkj.jtpdms.entity.system.LogInfo; importcom.zhkj.jtpdms.entity.system.User; importcom.zhkj.jtpdms.service.system.LogInfoService; importcom.zhkj.jtpdms.util.GlobalConstant; /** * 操作日志类 * * @author MEIM * */@Aspect @Component public classArchivesLogAspect { @Autowired privateLogInfoService loginfoService; private static final Logger logger = LoggerFactory.getLogger(ArchivesLog.class); @Pointcut(“@annotation(ArchivesLog)”) public voidcontrollerAspect() { //System.out.println(“切入点…”); } /** * 方法调用后触发 , 记录正常操作 * * @param joinPoint * @throws ClassNotFoundException */@AfterReturning(“controllerAspect()”) public void after(JoinPoint joinPoint) throwsClassNotFoundException { // 用户id int userId =getUSerMsg().getId(); // 用户IP String ip =getUSerMsg().getLoginIp(); // 控制器名 String targetName =getMethodDesc(joinPoint).getController(); // 方法名 String methodName =getMethodDesc(joinPoint).getMethod(); // 操作说明 String operteContent =getMethodDesc(joinPoint).getOperateContent(); LogInfo logInfo = newLogInfo(); logInfo.setUserId(userId); logInfo.setIp(ip); logInfo.setOperateContent(operteContent); logInfo.setMethod(methodName); logInfo.setController(targetName); loginfoService.insertLog(logInfo); } /** * 发生异常,走此方法 * @param joinPoint * @param e */@AfterThrowing(pointcut = “controllerAspect()”, throwing = “e”) public voidAfterThrowing(JoinPoint joinPoint, Throwable e) { try{ // 用户id int userId =getUSerMsg().getId(); // 用户IP String ip =getUSerMsg().getLoginIp(); // 控制器名 String targetName =getMethodDesc(joinPoint).getController(); // 方法名 String methodName =getMethodDesc(joinPoint).getMethod(); // 操作说明 String operteContent =getMethodDesc(joinPoint).getOperateContent(); LogInfo logInfo = newLogInfo(); String exMsg =e.getCause().toString(); if (exMsg != null) { int type = 2; logInfo.setUserId(userId); logInfo.setIp(ip); logInfo.setOperateContent(operteContent); logInfo.setMethod(methodName); logInfo.setController(targetName); logInfo.setType(type); logInfo.setExMsg(exMsg); loginfoService.insertLog(logInfo); } } catch(Exception e1) { logger.error(e1.getMessage()); } } /** * 获取 注解中对方法的描述 * * @return * @throws ClassNotFoundException */ public static LogInfo getMethodDesc(JoinPoint joinPoint) throwsClassNotFoundException { String targetName =joinPoint.getTarget().getClass().getName(); String methodName =joinPoint.getSignature().getName(); Object[] arguments =joinPoint.getArgs(); Class targetClass =Class.forName(targetName); Method[] methods =targetClass.getMethods(); String operteContent = “”; for(Method method : methods) { if(method.getName().equals(methodName)) { Class[] clazzs =method.getParameterTypes(); if (clazzs.length ==arguments.length) {
// 操作说明 operteContent = method.getAnnotation(ArchivesLog.class).operteContent(); break; } } } LogInfo logInfo = newLogInfo(); logInfo.setController(targetName); logInfo.setMethod(methodName); logInfo.setOperateContent(operteContent); returnlogInfo; } /** * 得到用户信息 * * @return */ public staticUser getUSerMsg() { HttpServletRequest req =((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest(); // 获取session HttpSession session =req.getSession(); User user =(User) session.getAttribute(GlobalConstant.SESSION_USER); returnuser; } }
5、在控制层/服务层加上自定义注解即可成功
@RequestMapping(“/saveUser”)
@ResponseBody
// 此处加上自定义注解 ,operteContent 是操作说明
@ArchivesLog(operteContent = “新增用户”)
public JsonResult saveUser(User user, String check) {
//… 增加操作
}
6、查看效果如下:
7、注意事项:
需要在spring-mvc 配置aop !!!
通知的几个属性(@around , @before ,@after ,@afterRetruning,@afterThrowing)需要弄清楚!!
还有不明白之处欢迎沟通!