Spring Boot利用AOP獲取用戶操作實(shí)現(xiàn)日志記錄

環(huán)境:IDEA版本2017.3.1 x64, JDK1.8蒲赂, SpringBoot2.1.1州丹, Druid1.1.8醋安, mybatis1.3.2杂彭,Security5.1.2,thymeleaf3.0.11

思路總結(jié):首先在需要做日志記錄的方法中添加一個(gè)自定義注解吓揪,再去實(shí)現(xiàn)一個(gè)日志AOP類亲怠,AOP類把自定義注解設(shè)置為切點(diǎn),所以當(dāng)系統(tǒng)執(zhí)行某一個(gè)添加了自定義注解的方法時(shí)柠辞,AOP會(huì)自動(dòng)獲取該方法名稱以及用戶信息實(shí)現(xiàn)日志記錄团秽。

編寫自定義注解

/**
 * 自定義注解類  定義controller方法的中文含義
 * @Target({METHOD,TYPE}) 表示這個(gè)注解可以用用在類/接口上,還可以用在方法上
 * @Retention(RetentionPolicy.RUNTIME) 表示這是一個(gè)運(yùn)行時(shí)注解叭首,即運(yùn)行起來(lái)之后习勤,才獲取注解中的相關(guān)信息,而不像基本注解如@Override 那種不用運(yùn)行焙格,在編譯時(shí)eclipse就可以進(jìn)行相關(guān)工作的編譯時(shí)注解图毕。
 * @Inherited 表示這個(gè)注解可以被子類繼承
 * @Documented 表示當(dāng)執(zhí)行javadoc的時(shí)候,本注解會(huì)生成相關(guān)文檔
 */

@Target({METHOD, TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface Operation {
    String value() default "";
}

日志AOP類會(huì)用到獲取IP地址的工具類眷唉,也用到j(luò)son工具類予颤,實(shí)現(xiàn)工具類如下

實(shí)現(xiàn)必要工具類

獲取ip地址工具類
        /**
     * 獲取用戶真實(shí)的ip地址
     * @param request
     * @return
     */
public class IpAdrressUtil {
    
    public static String getIpAdrress(HttpServletRequest request) {
            String ip = null;

            //X-Forwarded-For:Squid 服務(wù)代理
            String ipAddresses = request.getHeader("X-Forwarded-For");
            String unknown = "unknown";
            if (ipAddresses == null || ipAddresses.length() == 0 || unknown.equalsIgnoreCase(ipAddresses)) {
                //Proxy-Client-IP:apache 服務(wù)代理
                ipAddresses = request.getHeader("Proxy-Client-IP");
            }

            if (ipAddresses == null || ipAddresses.length() == 0 || unknown.equalsIgnoreCase(ipAddresses)) {
                //WL-Proxy-Client-IP:weblogic 服務(wù)代理
                ipAddresses = request.getHeader("WL-Proxy-Client-IP");
            }

            if (ipAddresses == null || ipAddresses.length() == 0 || unknown.equalsIgnoreCase(ipAddresses)) {
                //HTTP_CLIENT_IP:有些代理服務(wù)器
                ipAddresses = request.getHeader("HTTP_CLIENT_IP");
            }

            if (ipAddresses == null || ipAddresses.length() == 0 || unknown.equalsIgnoreCase(ipAddresses)) {
                //X-Real-IP:nginx服務(wù)代理
                ipAddresses = request.getHeader("X-Real-IP");
            }

            //有些網(wǎng)絡(luò)通過(guò)多層代理,那么獲取到的ip就會(huì)有多個(gè)冬阳,一般都是通過(guò)逗號(hào)(,)分割開來(lái)蛤虐,并且第一個(gè)ip為客戶端的真實(shí)IP
            if (ipAddresses != null && ipAddresses.length() != 0) {
                ip = ipAddresses.split(",")[0];
            }

            //還是不能獲取到,最后再通過(guò)request.getRemoteAddr();獲取
            if (ip == null || ip.length() == 0 || unknown.equalsIgnoreCase(ipAddresses)) {
                ip = request.getRemoteAddr();
            }
            return ip;
      }
    
    
}

實(shí)現(xiàn)json工具類
public class JacksonUtil {

    private final static ObjectMapper objectMapper = new ObjectMapper();

    private JacksonUtil() {

    }

    public static ObjectMapper getInstance() {
        return objectMapper;
    }

    /**
     * javaBean肝陪、列表數(shù)組轉(zhuǎn)換為json字符串
     */
    public static String obj2json(Object obj) throws Exception {
        return objectMapper.writeValueAsString(obj);
    }

    /**
     * json 轉(zhuǎn)JavaBean
     */

    public static <T> T json2pojo(String jsonString, Class<T> clazz) throws Exception {
        objectMapper.configure(DeserializationFeature.ACCEPT_SINGLE_VALUE_AS_ARRAY, true);
        return objectMapper.readValue(jsonString, clazz);
    }

    /**
     * json字符串轉(zhuǎn)換為map
     */
    public static <T> Map<String, Object> json2map(String jsonString) throws Exception {
        ObjectMapper mapper = new ObjectMapper();
        mapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
        return mapper.readValue(jsonString, Map.class);
    }
}

實(shí)現(xiàn)日志AOP類

/**
 * 系統(tǒng)日志:切面處理類
 */
@Aspect
@Component
public class SysLogAspect {

    @Autowired
    private SysLogService sysLogService;

    //定義切點(diǎn) @Pointcut
    //在注解的位置切入代碼
    @Pointcut("@annotation(cn.springboot.util.Operation)")
    public void logPoinCut() {
    }

    //切面 配置通知
    @AfterReturning("logPoinCut()")
    public void saveSysLog(JoinPoint joinPoint) {
        //保存日志
        SysLog sysLog = new SysLog();

        //從切面織入點(diǎn)處通過(guò)反射機(jī)制獲取織入點(diǎn)處的方法
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        //獲取切入點(diǎn)所在的方法
        Method method = signature.getMethod();


        //獲取操作
        Operation operation = method.getAnnotation(Operation.class);
        if (operation != null) {
            String value = operation.value();
            sysLog.setOperation(value);//保存獲取的操作
        }

        //獲取請(qǐng)求的類名
        String className = joinPoint.getTarget().getClass().getName();

        //獲取請(qǐng)求的方法名
        String methodName = method.getName();
        sysLog.setMethod(className + "." + methodName);

        //請(qǐng)求的參數(shù)
        Object[] args = joinPoint.getArgs();
        //將參數(shù)所在的數(shù)組轉(zhuǎn)換成json
        String params = null;
        try {
            params = JacksonUtil.obj2json(args);
        } catch (Exception e) {
            e.printStackTrace();
        }
        sysLog.setParams(params);

        //請(qǐng)求的時(shí)間
        sysLog.setCreateDate(new Date());

        //獲取用戶名
        Authentication authentication = SecurityContextHolder.getContext().getAuthentication();
        if (!(authentication instanceof AnonymousAuthenticationToken)) {
            sysLog.setUsername(authentication.getName());
        }

        //獲取用戶ip地址
        HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes())
                .getRequest();
        sysLog.setIp(IpAdrressUtil.getIpAdrress(request));

        //調(diào)用service保存SysLog實(shí)體類到數(shù)據(jù)庫(kù)
        sysLogService.saveLog(sysLog);
    }

}

最后驳庭,我是調(diào)用service保存SysLog實(shí)體類到數(shù)據(jù)庫(kù),你也可以直接輸出到控制臺(tái)氯窍,要保存到數(shù)據(jù)庫(kù)饲常,還要實(shí)現(xiàn)service類,mapper類和javeBean荞驴。
我就簡(jiǎn)單貼個(gè)javaBean不皆,其他類就不具體貼出了。

@Data
@NoArgsConstructor
@AllArgsConstructor
public class SysLog implements Serializable {
    private Long id;

    private String username; //用戶名

    private String operation; //操作

    private String method; //方法名

    private String params; //參數(shù)

    private String ip; //ip地址

    private Date createDate; //操作時(shí)間
    //創(chuàng)建getter和setter方法
}

javaBean中使用lombok的注解實(shí)現(xiàn)了get熊楼、set等方法霹娄。

以下是存進(jìn)數(shù)據(jù)庫(kù)的數(shù)據(jù),關(guān)于IP的問題是因?yàn)楸镜販y(cè)試鲫骗,部署在服務(wù)器上就會(huì)有正常的IP地址了犬耻。

image

更多Spring Boot整合可瀏覽此博客:malizhi.cn

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市执泰,隨后出現(xiàn)的幾起案子枕磁,更是在濱河造成了極大的恐慌,老刑警劉巖术吝,帶你破解...
    沈念sama閱讀 212,884評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件计济,死亡現(xiàn)場(chǎng)離奇詭異茸苇,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)沦寂,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,755評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門学密,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人传藏,你說(shuō)我怎么就攤上這事腻暮。” “怎么了毯侦?”我有些...
    開封第一講書人閱讀 158,369評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵哭靖,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我侈离,道長(zhǎng)试幽,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,799評(píng)論 1 285
  • 正文 為了忘掉前任卦碾,我火速辦了婚禮抡草,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘蔗坯。我一直安慰自己,他們只是感情好燎含,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,910評(píng)論 6 386
  • 文/花漫 我一把揭開白布宾濒。 她就那樣靜靜地躺著,像睡著了一般屏箍。 火紅的嫁衣襯著肌膚如雪绘梦。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 50,096評(píng)論 1 291
  • 那天赴魁,我揣著相機(jī)與錄音卸奉,去河邊找鬼。 笑死颖御,一個(gè)胖子當(dāng)著我的面吹牛榄棵,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播潘拱,決...
    沈念sama閱讀 39,159評(píng)論 3 411
  • 文/蒼蘭香墨 我猛地睜開眼疹鳄,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了芦岂?” 一聲冷哼從身側(cè)響起瘪弓,我...
    開封第一講書人閱讀 37,917評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎禽最,沒想到半個(gè)月后腺怯,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體袱饭,經(jīng)...
    沈念sama閱讀 44,360評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,673評(píng)論 2 327
  • 正文 我和宋清朗相戀三年呛占,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了虑乖。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,814評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡栓票,死狀恐怖决左,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情走贪,我是刑警寧澤佛猛,帶...
    沈念sama閱讀 34,509評(píng)論 4 334
  • 正文 年R本政府宣布,位于F島的核電站坠狡,受9級(jí)特大地震影響继找,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜逃沿,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,156評(píng)論 3 317
  • 文/蒙蒙 一婴渡、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧凯亮,春花似錦边臼、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,882評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至富拗,卻和暖如春臼予,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背啃沪。 一陣腳步聲響...
    開封第一講書人閱讀 32,123評(píng)論 1 267
  • 我被黑心中介騙來(lái)泰國(guó)打工粘拾, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人创千。 一個(gè)月前我還...
    沈念sama閱讀 46,641評(píng)論 2 362
  • 正文 我出身青樓缰雇,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親追驴。 傳聞我的和親對(duì)象是個(gè)殘疾皇子寓涨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,728評(píng)論 2 351

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