Spring AOP實(shí)現(xiàn)后臺(tái)管理系統(tǒng)日志管理

元注解方式結(jié)合AOP施逾,靈活記錄操作日志
能夠記錄詳細(xì)錯(cuò)誤日志為運(yùn)維提供支持
日志記錄盡可能減少性能影響

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 {
    /**
     * 描述業(yè)務(wù)操作 例:Xxx管理-執(zhí)行Xxx操作
     * @return
     */
    String description() default "";
}

2.定義用于記錄日志的實(shí)體類

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;           //日志標(biāo)題  
    private String remoteAddr;          //請(qǐng)求地址  
    private String requestUri;          //URI   
    private String method;          //請(qǐng)求方式  
    private String params;          //提交參數(shù)  
    private String exception;           //異常    
    @JsonFormat(pattern="yyyy-MM-dd HH:mm:ss", timezone = "GMT+8")
    private Date operateDate;           //開始時(shí)間  
    private String timeout;         //結(jié)束時(shí)間  
    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;
    }

    /**
     * 設(shè)置請(qǐng)求參數(shù)
     * @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;

/**
* 系統(tǒng)日志切面類
* @author lin.r.x
*
*/
@Aspect
@Component
public 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層切點(diǎn) 注解攔截
    */
   @Pointcut("@annotation(com.myron.ims.annotation.SystemControllerLog)")
   public void controllerAspect(){}

   /**
    * 前置通知 用于攔截Controller層記錄用戶的操作的開始時(shí)間
    * @param joinPoint 切點(diǎn)
    * @throws InterruptedException 
    */
   @Before("controllerAspect()")
   public void doBefore(JoinPoint joinPoint) throws InterruptedException{
       Date beginTime=new Date();
       beginTimeThreadLocal.set(beginTime);//線程綁定變量(該數(shù)據(jù)只有當(dāng)前請(qǐng)求的線程可見)  
       if (logger.isDebugEnabled()){//這里日志級(jí)別為debug
           logger.debug("開始計(jì)時(shí): {}  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 切點(diǎn)
    */
   @SuppressWarnings("unchecked")
   @After("controllerAspect()")
   public void doAfter(JoinPoint joinPoint) {
       User user = currentUser.get();
       if(user !=null){
           String title="";
           String type="info";                       //日志類型(info:入庫,error:錯(cuò)誤)
           String remoteAddr=request.getRemoteAddr();//請(qǐng)求的IP
           String requestUri=request.getRequestURI();//請(qǐng)求的Uri
           String method=request.getMethod();        //請(qǐng)求的方法類型(post/get)
           Map<String,String[]> params=request.getParameterMap(); //請(qǐng)求提交的參數(shù)

           try {
               title=getControllerMethodDescription2(joinPoint);
           } catch (Exception e) {
               e.printStackTrace();
           }    
           // 打印JVM信息。
           long beginTime = beginTimeThreadLocal.get().getTime();//得到線程綁定的局部變量(開始時(shí)間)  
           long endTime = System.currentTimeMillis();  //2桐绒、結(jié)束時(shí)間  
           if (logger.isDebugEnabled()){
               logger.debug("計(jì)時(shí)結(jié)束:{}  URI: {}  耗時(shí): {}   最大內(nèi)存: {}m  已分配內(nèi)存: {}m  已分配內(nèi)存中的剩余空間: {}m  最大可用內(nèi)存: {}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.直接執(zhí)行保存操作
           //this.logService.createSystemLog(log);

           //2.優(yōu)化:異步保存日志
           //new SaveLogThread(log, logService).start();

           //3.再優(yōu)化:通過線程池來執(zhí)行日志保存
           threadPoolTaskExecutor.execute(new SaveLogThread(log, logService));
           logThreadLocal.set(log);
       }

   }

   /**
    *  異常通知 記錄操作報(bào)錯(cuò)日志
    * @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();
   }

   /**
    * 獲取注解中對(duì)方法的描述信息 用于service層注解
    * @param joinPoint切點(diǎn)
    * @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;
   }

   /**
    * 獲取注解中對(duì)方法的描述信息 用于Controller層注解
    * 
    * @param joinPoint 切點(diǎn)
    * @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注解的支持

···

<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方法中添加日志注解

···
/**
*系統(tǒng)登入
*/
@RequestMapping("/login.do")
@SystemControllerLog(description="登入系統(tǒng)")
@ResponseBody
public Map<String, Object> login(String username, String password, Boolean rememberMe, HttpServletRequest req){
//業(yè)務(wù)代碼省略...
}

/**
 * 安全退出登入
 * @return
 */
@SystemControllerLog(description="安全退出系統(tǒng)")
@RequestMapping("logout.do")
public String logout(){
    Subject subject=SecurityUtils.getSubject();
    if(subject.isAuthenticated()){
        subject.logout(); // session 會(huì)銷毀糕再,在SessionListener監(jiān)聽session銷毀锨能,清理權(quán)限緩存
    }
    return "/login.jsp";
}

6.運(yùn)行效果


20170208164120917.jpg
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市传黄,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌队寇,老刑警劉巖膘掰,帶你破解...
    沈念sama閱讀 218,122評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異佳遣,居然都是意外死亡识埋,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,070評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門苍日,熙熙樓的掌柜王于貴愁眉苦臉地迎上來惭聂,“玉大人,你說我怎么就攤上這事相恃」几伲” “怎么了?”我有些...
    開封第一講書人閱讀 164,491評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵拦耐,是天一觀的道長(zhǎng)耕腾。 經(jīng)常有香客問我,道長(zhǎng)杀糯,這世上最難降的妖魔是什么扫俺? 我笑而不...
    開封第一講書人閱讀 58,636評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮固翰,結(jié)果婚禮上狼纬,老公的妹妹穿的比我還像新娘。我一直安慰自己骂际,他們只是感情好疗琉,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,676評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著歉铝,像睡著了一般盈简。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上太示,一...
    開封第一講書人閱讀 51,541評(píng)論 1 305
  • 那天柠贤,我揣著相機(jī)與錄音,去河邊找鬼类缤。 笑死臼勉,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的餐弱。 我是一名探鬼主播坚俗,決...
    沈念sama閱讀 40,292評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼镜盯,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了猖败?” 一聲冷哼從身側(cè)響起速缆,我...
    開封第一講書人閱讀 39,211評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎恩闻,沒想到半個(gè)月后艺糜,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,655評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡幢尚,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,846評(píng)論 3 336
  • 正文 我和宋清朗相戀三年破停,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片尉剩。...
    茶點(diǎn)故事閱讀 39,965評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡真慢,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出理茎,到底是詐尸還是另有隱情黑界,我是刑警寧澤,帶...
    沈念sama閱讀 35,684評(píng)論 5 347
  • 正文 年R本政府宣布皂林,位于F島的核電站朗鸠,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏础倍。R本人自食惡果不足惜烛占,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,295評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望沟启。 院中可真熱鬧忆家,春花似錦、人聲如沸德迹。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,894評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽浦辨。三九已至,卻和暖如春沼沈,著一層夾襖步出監(jiān)牢的瞬間流酬,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,012評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工列另, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留芽腾,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,126評(píng)論 3 370
  • 正文 我出身青樓页衙,卻偏偏與公主長(zhǎng)得像摊滔,于是被迫代替她去往敵國(guó)和親阴绢。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,914評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容