亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 操作系統 > 正文

Spring AOP實現后臺管理系統日志管理

2024-06-28 16:01:45
字體:
來源:轉載
供稿:網友

SPRing AOP實現后臺管理系統日志管理

設計原則和思路:

元注解方式結合AOP,靈活記錄操作日志能夠記錄詳細錯誤日志為運維提供支持日志記錄盡可能減少性能影響

1.定義日志記錄元注解

package com.myron.ims.annotation;import java.lang.annotation.*;/** * 自定義注解 攔截Controller * * @author lin.r.x * */@Target({ ElementType.PARAMETER, ElementType.METHOD })@Retention(RetentionPolicy.RUNTIME)public @interface SystemControllerLog { /** * 描述業務操作 例:Xxx管理-執行Xxx操作 * @return */ String description() default "";}

2.定義用于記錄日志的實體類

package com.myron.ims.bean;import java.io.Serializable;import com.myron.common.util.StringUtils;import com.myron.common.util.UuidUtils;import com.fasterxml.jackson.annotation.JsonFormat;import java.util.Date;import java.util.Map;/** * 日志類-記錄用戶操作行為 * @author lin.r.x * */public class Log implements Serializable{ private static final long serialVersionUID = 1L; private String logId; //日志主鍵 private String type; //日志類型 private String title; //日志標題 private String remoteAddr; //請求地址 private String requestUri; //URI private String method; //請求方式 private String params; //提交參數 private String exception; //異常 @JsonFormat(pattern="yyyy-MM-dd HH:mm:ss", timezone = "GMT+8") private Date OperateDate; //開始時間 private String timeout; //結束時間 private String userId; //用戶ID public String getLogId() { return StringUtils.isBlank(logId) ? logId : logId.trim(); } public void setLogId(String logId) { this.logId = logId; } public String getType() { return StringUtils.isBlank(type) ? type : type.trim(); } public void setType(String type) { this.type = type; } public String getTitle() { return StringUtils.isBlank(title) ? title : title.trim(); } public void setTitle(String title) { this.title = title; } public String getRemoteAddr() { return StringUtils.isBlank(remoteAddr) ? remoteAddr : remoteAddr.trim(); } public void setRemoteAddr(String remoteAddr) { this.remoteAddr = remoteAddr; } public String getRequestUri() { return StringUtils.isBlank(requestUri) ? requestUri : requestUri.trim(); } public void setRequestUri(String requestUri) { this.requestUri = requestUri; } public String getMethod() { return StringUtils.isBlank(method) ? method : method.trim(); } public void setMethod(String method) { this.method = method; } public String getParams() { return StringUtils.isBlank(params) ? params : params.trim(); } public void setParams(String params) { this.params = params; } /** * 設置請求參數 * @param paramMap */ public void setMapToParams(Map<String, String[]> paramMap) { if (paramMap == null){ return; } StringBuilder params = new StringBuilder(); for (Map.Entry<String, String[]> param : ((Map<String, String[]>)paramMap).entrySet()){ params.append(("".equals(params.toString()) ? "" : "&") + param.getKey() + "="); String paramValue = (param.getValue() != null && param.getValue().length > 0 ? param.getValue()[0] : ""); params.append(StringUtils.abbr(StringUtils.endsWithIgnoreCase(param.getKey(), "passWord") ? "" : paramValue, 100)); } this.params = params.toString(); } public String getException() { return StringUtils.isBlank(exception) ? exception : exception.trim(); } public void setException(String exception) { this.exception = exception; } public Date getOperateDate() { return operateDate; } public void setOperateDate(Date operateDate) { this.operateDate = operateDate; } public String getTimeout() { return StringUtils.isBlank(timeout) ? timeout : timeout.trim(); } public void setTimeout(String timeout) { this.timeout = timeout; } public String getUserId() { return StringUtils.isBlank(userId) ? userId : userId.trim(); } public void setUserId(String userId) { this.userId = userId; }}

3.定義日志AOP切面類

package com.myron.ims.aop;import java.lang.reflect.Method;import java.text.SimpleDateFormat;import java.util.Date;import java.util.Map;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.Httpsession;import org.aspectj.lang.JoinPoint;import org.aspectj.lang.annotation.After;import org.aspectj.lang.annotation.AfterThrowing;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.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.core.NamedThreadLocal;import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;import org.springframework.stereotype.Component;import com.myron.common.util.DateUtils;import com.myron.common.util.UuidUtils;import com.myron.ims.annotation.SystemControllerLog;import com.myron.ims.annotation.SystemServiceLog;import com.myron.ims.bean.Log;import com.myron.ims.bean.User;import com.myron.ims.service.LogService;/** * 系統日志切面類 * @author lin.r.x * */@Aspect@Componentpublic class SystemLogAspect { private static final Logger logger = LoggerFactory.getLogger(SystemLogAspect. class); private static final ThreadLocal<Date> beginTimeThreadLocal = new NamedThreadLocal<Date>("ThreadLocal beginTime"); private static final ThreadLocal<Log> logThreadLocal = new NamedThreadLocal<Log>("ThreadLocal log"); private static final ThreadLocal<User> currentUser=new NamedThreadLocal<>("ThreadLocal user"); @Autowired(required=false) private HttpServletRequest request; @Autowired private ThreadPoolTaskExecutor threadPoolTaskExecutor; @Autowired private LogService logService; /** * Controller層切點 注解攔截 */ @Pointcut("@annotation(com.myron.ims.annotation.SystemControllerLog)") public void controllerAspect(){} /** * 前置通知 用于攔截Controller層記錄用戶的操作的開始時間 * @param joinPoint 切點 * @throws InterruptedException */ @Before("controllerAspect()") public void doBefore(JoinPoint joinPoint) throws InterruptedException{ Date beginTime=new Date(); beginTimeThreadLocal.set(beginTime);//線程綁定變量(該數據只有當前請求的線程可見) if (logger.isDebugEnabled()){//這里日志級別為debug logger.debug("開始計時: {} URI: {}", new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS") .format(beginTime), request.getRequestURI()); } //讀取session中的用戶 HttpSession session = request.getSession(); User user = (User) session.getAttribute("ims_user"); currentUser.set(user); } /** * 后置通知 用于攔截Controller層記錄用戶的操作 * @param joinPoint 切點 */ @SuppressWarnings("unchecked") @After("controllerAspect()") public void doAfter(JoinPoint joinPoint) { User user = currentUser.get(); if(user !=null){ String title=""; String type="info"; //日志類型(info:入庫,error:錯誤) String remoteAddr=request.getRemoteAddr();//請求的IP String requestUri=request.getRequestURI();//請求的Uri String method=request.getMethod(); //請求的方法類型(post/get) Map<String,String[]> params=request.getParameterMap(); //請求提交的參數 try { title=getControllerMethodDescription2(joinPoint); } catch (Exception e) { e.printStackTrace(); } // 打印JVM信息。 long beginTime = beginTimeThreadLocal.get().getTime();//得到線程綁定的局部變量(開始時間) long endTime = System.currentTimeMillis(); //2、結束時間 if (logger.isDebugEnabled()){ logger.debug("計時結束:{} URI: {} 耗時: {} 最大內存: {}m 已分配內存: {}m 已分配內存中的剩余空間: {}m 最大可用內存: {}m", new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS").format(endTime), request.getRequestURI(), DateUtils.formatDateTime(endTime - beginTime), Runtime.getRuntime().maxMemory()/1024/1024, Runtime.getRuntime().totalMemory()/1024/1024, Runtime.getRuntime().freeMemory()/1024/1024, (Runtime.getRuntime().maxMemory()-Runtime.getRuntime().totalMemory()+Runtime.getRuntime().freeMemory())/1024/1024); } Log log=new Log(); log.setLogId(UuidUtils.creatUUID()); log.setTitle(title); log.setType(type); log.setRemoteAddr(remoteAddr); log.setRequestUri(requestUri); log.setMethod(method); log.setMapToParams(params); log.setUserId(user.getId()); Date operateDate=beginTimeThreadLocal.get(); log.setOperateDate(operateDate); log.setTimeout(DateUtils.formatDateTime(endTime - beginTime)); //1.直接執行保存操作 //this.logService.createSystemLog(log); //2.優化:異步保存日志 //new SaveLogThread(log, logService).start(); //3.再優化:通過線程池來執行日志保存 threadPoolTaskExecutor.execute(new SaveLogThread(log, logService)); logThreadLocal.set(log); } } /** * 異常通知 記錄操作報錯日志 * @param joinPoint * @param e */ @AfterThrowing(pointcut = "controllerAspect()", throwing = "e") public void doAfterThrowing(JoinPoint joinPoint, Throwable e) { Log log = logThreadLocal.get(); log.setType("error"); log.setException(e.toString()); new UpdateLogThread(log, logService).start(); } /** * 獲取注解中對方法的描述信息 用于service層注解 * @param joinPoint切點 * @return discription */ public static String getServiceMthodDescription2(JoinPoint joinPoint) { MethodSignature signature = (MethodSignature) joinPoint.getSignature(); Method method = signature.getMethod(); SystemServiceLog serviceLog = method .getAnnotation(SystemServiceLog.class); String discription = serviceLog.description(); return discription; } /** * 獲取注解中對方法的描述信息 用于Controller層注解 * * @param joinPoint 切點 * @return discription */ public static String getControllerMethodDescription2(JoinPoint joinPoint) { MethodSignature signature = (MethodSignature) joinPoint.getSignature(); Method method = signature.getMethod(); SystemControllerLog controllerLog = method .getAnnotation(SystemControllerLog.class); String discription = controllerLog.description(); return discription; } /** * 保存日志線程 */ private static class SaveLogThread implements Runnable { private Log log; private LogService logService; public SaveLogThread(Log log, LogService logService) { this.log = log; this.logService = logService; } @Override public void run() { logService.createLog(log); } } /** * 日志更新線程 */ private static class UpdateLogThread extends Thread { private Log log; private LogService logService; public UpdateLogThread(Log log, LogService logService) { super(UpdateLogThread.class.getSimpleName()); this.log = log; this.logService = logService; } @Override public void run() { this.logService.updateLog(log); } }}

4.spring 配置掃描切面,開啟@AspectJ注解的支持

<!-- 啟動對@AspectJ注解的支持 --> <aop:aspectj-autoproxy/> <!-- 掃描切點類組件 --> <context:component-scan base-package="com.myron.ims.aop" /> <context:component-scan base-package="com.myron.ims.service"/> <bean id="taskExecutor" class="org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor"> <property name="corePoolSize" value="5" /> <property name="maxPoolSize" value="10" /> <property name="WaitForTasksToCompleteOnShutdown" value="true" /> </bean>

5.使用范例LoginController方法中添加日志注解

/** *系統登入 */ @RequestMapping("/login.do") @SystemControllerLog(description="登入系統") @ResponseBody public Map<String, Object> login(String username, String password, Boolean rememberMe, HttpServletRequest req){ //業務代碼省略...} /** * 安全退出登入 * @return */ @SystemControllerLog(description="安全退出系統") @RequestMapping("logout.do") public String logout(){ Subject subject=SecurityUtils.getSubject(); if(subject.isAuthenticated()){ subject.logout(); // session 會銷毀,在SessionListener監聽session銷毀,清理權限緩存 } return "/login.jsp"; }

6.運行效果 這里寫圖片描述


上一篇:openstack-M版--Compute端

下一篇:I2C子系統

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
午夜精品一区二区三区在线| 中文字幕欧美日韩在线| 性金发美女69hd大尺寸| 亚洲欧美中文日韩在线| 亚洲日本aⅴ片在线观看香蕉| 成人日韩在线电影| 久久久久久国产精品久久| 久久精品国产清自在天天线| 91精品国产91久久久久福利| 国产在线拍揄自揄视频不卡99| 国产日韩精品在线播放| 日韩精品中文字幕久久臀| 久久久久久久999精品视频| 欧美视频一二三| 亚洲丁香婷深爱综合| 国产一区二区三区在线看| 久久久999精品免费| 插插插亚洲综合网| 欧美小视频在线| 91精品久久久久久久久| 中文字幕日韩欧美| 大量国产精品视频| 夜夜嗨av色一区二区不卡| 97涩涩爰在线观看亚洲| 国产精品久久久久久久久久久久| 欧美亚洲成人精品| 911国产网站尤物在线观看| 91亚洲va在线va天堂va国| 欧美乱妇高清无乱码| 成人xvideos免费视频| 国产精品户外野外| 97视频在线观看免费高清完整版在线观看| 91亚洲精品在线| www.欧美精品一二三区| 欧美精品18videosex性欧美| 国产91久久婷婷一区二区| 91国语精品自产拍在线观看性色| 麻豆国产精品va在线观看不卡| 精品国产精品自拍| 亚洲免费视频一区二区| 亚洲精品国产精品自产a区红杏吧| 久久久久久亚洲精品不卡| 欧美激情在线视频二区| 欧美成人午夜剧场免费观看| 黄色91在线观看| 亚洲影院色在线观看免费| 国产精品免费视频久久久| 国语对白做受69| 欧美日韩裸体免费视频| 午夜精品久久久久久久久久久久| 亚洲欧美资源在线| 欧美性猛交xxxx免费看久久久| 亚洲精品一区二区三区不| 97国产精品视频人人做人人爱| 尤物yw午夜国产精品视频| 欧美国产日韩精品| 久国内精品在线| 欧美成人激情图片网| 91精品国产乱码久久久久久久久| 伊人男人综合视频网| 日韩av男人的天堂| 91国内精品久久| 国产一区二区日韩精品欧美精品| 欧美成aaa人片在线观看蜜臀| 亚洲日本中文字幕免费在线不卡| 欧美激情国产高清| 久久精品这里热有精品| 日韩成人中文电影| 国产欧美在线观看| 欧美最顶级丰满的aⅴ艳星| 91高清免费视频| 亚洲电影免费观看高清| 欧美午夜精品久久久久久浪潮| 亚洲人成电影网站| 影音先锋欧美精品| 久久网福利资源网站| 欧美成人免费观看| www欧美日韩| 日本国产高清不卡| 91久久精品国产91久久性色| 国产精品白丝av嫩草影院| 久久精品视频导航| 欧美一区二区三区四区在线| 日韩精品视频免费在线观看| 亚洲欧美国产精品专区久久| 中文字幕日本精品| 国产精品高潮视频| 亚洲欧美在线磁力| 国产亚洲精品美女久久久| 国产亚洲欧美另类中文| 欧美日韩精品在线播放| 久久亚洲精品一区| 欧美精品videos| 国产97色在线| www欧美日韩| 欧美老妇交乱视频| 国产91露脸中文字幕在线| 国产成人精品免高潮在线观看| 自拍亚洲一区欧美另类| 日本三级韩国三级久久| 国产日韩在线精品av| 久久精彩免费视频| 亚洲性av在线| 欧美人交a欧美精品| 欧美成人免费网| 亚洲xxxx18| 色综合久久精品亚洲国产| 日本三级韩国三级久久| 欧美电影免费观看电视剧大全| 97婷婷大伊香蕉精品视频| 久久久爽爽爽美女图片| 亚洲人成伊人成综合网久久久| 久久在精品线影院精品国产| 97人人爽人人喊人人模波多| 日韩在线视频国产| 欧美激情第6页| 国产伦精品一区二区三区精品视频| 日韩欧美高清在线视频| 久久欧美在线电影| 久久伊人精品视频| 红桃视频成人在线观看| 日韩av电影国产| 日本欧美一级片| 57pao国产精品一区| 国产精品99导航| 中文字幕精品一区二区精品| 欧美成在线观看| 伊人久久免费视频| 中文字幕日本精品| 国产在线999| 国产精品视频一| 亚洲国产成人在线视频| 国产精品欧美日韩一区二区| 亚洲欧美中文另类| 欧美成人午夜激情| 91亚洲精品在线观看| 91高清视频在线免费观看| 日韩黄色在线免费观看| 日本亚洲欧美三级| 日韩视频免费看| 国内精品模特av私拍在线观看| 亚洲黄色av网站| 欧美日韩中文字幕在线视频| 色妞一区二区三区| 国产一区二区丝袜高跟鞋图片| 国产成人av网址| 国产精品久久av| 2018日韩中文字幕| 国产99久久精品一区二区永久免费| 欧美电影免费观看网站| 国产欧美精品一区二区三区介绍| 久久久成人的性感天堂| 日韩av片免费在线观看| 一个色综合导航| 亚洲尤物视频网| 久久久亚洲影院| 92裸体在线视频网站| 欧美激情精品久久久久久黑人| 欧美成人精品在线观看| 亚洲区一区二区| 秋霞成人午夜鲁丝一区二区三区| 91精品久久久久久久久久入口| 亚洲欧美自拍一区| 亚洲国产精品成人va在线观看|