Spring如何基于aop實(shí)現(xiàn)操作日志功能
1. 在pom中添加所需依賴
創(chuàng)建一個(gè)springboot工程,添加所需要的依賴,持久化用的是mybatis
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!--springboot aop依賴-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
<!--mybatis-->
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>2.1.3</version>
</dependency>
<!--mysql連接-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.19</version>
<scope>runtime</scope>
</dependency>
<!--lombok-->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
2. 創(chuàng)建日志實(shí)體類
import lombok.Data;
import java.io.Serializable;
@Data
public class AdminLog implements Serializable {
private static final long serialVersionUID = -291495801959706565L;
private Integer id; //日志記錄id
private Integer userId;//操作人id
private String userName;//操作人name
private String loginip;//登錄ip
private int type;
private String url;
private String operation;
private String createtime;
private String remark;
}
3. 自定義log注解
import java.lang.annotation.*;
/**
* 自定義日志注解
*/
@Target(ElementType.METHOD) //注解防止位置
@Retention(RetentionPolicy.RUNTIME)//運(yùn)行時(shí)可見
@Documented //生成文檔
public @interface MyLog {
String operation() default "";
int type();
}
4. 創(chuàng)建aop切面處理類
import cn.***.springaopdemo.anno.MyLog;
import cn.***.springaopdemo.dao.MyLogMapper;
import cn.***.springaopdemo.pojo.Admin;
import cn.***.springaopdemo.pojo.AdminLog;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import javax.servlet.http.HttpServletRequest;
import java.lang.reflect.Method;
/**
* 切面處理類
*/
@Aspect
@Component
public class SysLogAspect {
/**
* 使用log4j2把一些信息打印在控制臺(tái)上面
*/
private static final Logger log = LogManager.getLogger(SysLogAspect.class);
@Autowired
private MyLogMapper myLogMapper;
//定義切點(diǎn) @Pointcut
//在注解的位置切入代碼
@Pointcut("@annotation(cn.***.springaopdemo.anno.MyLog)")
public void logPointCut() {
}
//切面 配置為前置通知
@Before("logPointCut()")
public void saveOperation(JoinPoint joinPoint) {
log.info("---------------接口日志記錄---------------");
//創(chuàng)建一個(gè)日志對象
AdminLog adminLog = new AdminLog();
//獲取切面織處入點(diǎn)的方法
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
//獲取切入點(diǎn)所在的方法
Method method = signature.getMethod();
//獲取操作日志的屬性值
MyLog myLog = method.getAnnotation(MyLog.class);
if (myLog != null) {
//操作事件
String operation = myLog.operation();
adminLog.setOperation(operation);
//日志類型
int type = myLog.type();
adminLog.setType(type);
log.info("operation=" + operation + ",type=" + type);
}
//獲取url
HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();
String requestURL = request.getRequestURI().toString();
adminLog.setUrl(requestURL);
//獲取客戶端ip
String ip = request.getRemoteAddr();
adminLog.setLoginip(ip);
//獲取操作人賬號(hào)、姓名(需要提前將用戶信息保存到Session)
Admin admin = (Admin) request.getSession().getAttribute("admin");
if (admin != null) {
Integer id = admin.getId();
String name = admin.getName();
adminLog.setUserId(id);
adminLog.setUserName(name);
}
log.info("url=" + requestURL + ",ip=" + ip);
//調(diào)用service保存Operation實(shí)體類到數(shù)據(jù)庫
//可以在這設(shè)置id,因?yàn)槭菧y試,這里就使用的是數(shù)據(jù)庫的自增id
myLogMapper.insertLog(adminLog);
}
}
5. mapper層把日志數(shù)據(jù)存儲(chǔ)到mysql數(shù)據(jù)庫中
mapper接口
import cn.***.springaopdemo.pojo.AdminLog;
import java.util.List;
public interface MyLogMapper {
void insertLog(AdminLog adminLog);
}
mapper.xml文件
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="cn.***.springaopdemo.dao.MyLogMapper">
<insert id="insertLog" parameterType="cn.***.springaopdemo.pojo.AdminLog">
INSERT INTO admin_log (user_id,user_name,loginip,type,url,operation,createtime,remark)
VALUES (#{userId},#{userName},#{loginip},#{type},#{url},#{operation},now(),#{remark})
</insert>
</mapper>
6. 測試
先直接登錄用戶,因?yàn)槭菧y試,直接從數(shù)據(jù)庫中獲取后登錄,把a(bǔ)dmin存儲(chǔ)到session中
import cn.***.springaopdemo.pojo.Admin;
import cn.***.springaopdemo.service.IAdminService;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletRequest;
import java.util.List;
@RestController
@RequestMapping("/admin")
public class AdminController {
private static final Logger log = LogManager.getLogger(AdminController.class);
//中間service層可以省略,直接通過mapper接口操作數(shù)據(jù)即可
@Autowired
private IAdminService adminService;
@RequestMapping("/login")
public Admin login(HttpServletRequest request) {
List<Admin> adminList = adminService.findAllAdmin();
Admin admin = adminList.get(0);
request.getSession().setAttribute("admin",admin );
return admin;
}
}
在瀏覽器中輸入localhost:8080/admin/login,可以看到登錄的admin

進(jìn)行插入和查詢操作,插入數(shù)據(jù)直接通過后臺(tái)提供
import cn.***.springaopdemo.anno.MyLog;
import cn.***.springaopdemo.pojo.Type;
import cn.***.springaopdemo.service.ITypeService;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.ArrayList;
import java.util.List;
@RestController
@RequestMapping("/type")
public class TypeController {
private static final Logger log = LogManager.getLogger(TypeController.class);
@Autowired
private ITypeService typeService;
@MyLog(operation = "增加書籍類型", type = 2)
@RequestMapping("/add")
public void insertType() {
List<Type> typeList = new ArrayList<>();
Type type = new Type();
type.setName("自然科學(xué)");
typeList.add(type);
typeService.addTypeList(typeList);
log.info("添加書籍類型" + type.getName());
}
@MyLog(operation = "查詢所有書籍類型", type = 1)
@RequestMapping("/findAll")
public List<Type> findAllType() {
List<Type> typeList = typeService.findAllType();
log.info("查詢所有書籍類型");
return typeList;
}
}
在瀏覽器中輸入localhost:8080/type/add,后臺(tái)日志打印記錄

再輸入查詢請求localhost:8080/type/findAll,獲得查詢出的分類

查看數(shù)據(jù)庫是否添加成功

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot學(xué)習(xí)篇之@Valid與@Validated的區(qū)別
@Valid是使用Hibernate?validation的時(shí)候使用,@Validated是只用Spring?Validator校驗(yàn)機(jī)制使用,下面這篇文章主要給大家介紹了關(guān)于SpringBoot學(xué)習(xí)篇之@Valid與@Validated區(qū)別的相關(guān)資料,需要的朋友可以參考下2022-11-11
Java Stream中的Spliterator類概念及原理解析
Spliterator是Java 8引入的一個(gè)接口,位于java.util包中,它結(jié)合了迭代器(Iterator)的遍歷能力和分割器(Splitter)的分割能力,本文將詳細(xì)介紹Spliterator的概念、原理、作用、類中定義的關(guān)鍵方法,以及它在Stream API中的實(shí)際應(yīng)用,感興趣的朋友一起看看吧2024-08-08
Java多線程實(shí)現(xiàn)Callable接口
本文給大家分享的是使用Java多線程來實(shí)現(xiàn)callable接口的方法,以及使用方法,另外還有一個(gè)網(wǎng)友的實(shí)例,希望能夠?qū)Υ蠹艺莆認(rèn)ava多線程有所幫助。2016-06-06
基于Spring BeanUtils的copyProperties方法使用及注意事項(xiàng)
這篇文章主要介紹了基于Spring BeanUtils的copyProperties方法使用及注意事項(xiàng),具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06
SpringBoot+SpringSecurity+JWT實(shí)現(xiàn)系統(tǒng)認(rèn)證與授權(quán)示例
本文主要介紹了SpringBoot+SpringSecurity+JWT實(shí)現(xiàn)系統(tǒng)認(rèn)證與授權(quán)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
如果淘寶的七天自動(dòng)確認(rèn)收貨讓你設(shè)計(jì)你用Java怎么實(shí)現(xiàn)
在面試的時(shí)候如果面試官問淘寶的七天自動(dòng)確認(rèn)收貨讓你設(shè)計(jì),你會(huì)怎么具體實(shí)現(xiàn)呢?跟著小編看一下下邊的實(shí)現(xiàn)過程,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值2021-09-09

