AOP實(shí)戰(zhàn)篇 如何輕松實(shí)現(xiàn)日志功能啼辣,戳這

前言

你是不是還在類里面直接拿到日志對(duì)象調(diào)用方法來輸出日志燕侠?接下來教你如何簡(jiǎn)單使用AOP+自定義注解優(yōu)雅的實(shí)現(xiàn)一個(gè)日志記錄功能

提示:以下是本篇文章正文內(nèi)容镇草,案例僅供參考

一厉萝、技術(shù)介紹

1.AOP是什么恍飘?

在軟件業(yè),AOP為Aspect Oriented Programming的縮寫谴垫,意為:面向切面編程章母,通過預(yù)編譯方式和運(yùn)行期間動(dòng)態(tài)代理實(shí)現(xiàn)程序功能的統(tǒng)一維護(hù)的一種技術(shù)。AOP是OOP的延續(xù)翩剪,是軟件開發(fā)中的一個(gè)熱點(diǎn)乳怎,也是Spring框架中的一個(gè)重要內(nèi)容,是函數(shù)式編程的一種衍生范型前弯。利用AOP可以對(duì)業(yè)務(wù)邏輯的各個(gè)部分進(jìn)行隔離蚪缀,從而使得業(yè)務(wù)邏輯各部分之間的耦合度降低,提高程序的可重用性恕出,同時(shí)提高了開發(fā)的效率询枚。 --摘自百度百科

二、開始使用

1.代碼目錄結(jié)構(gòu)

image.png

2.開始編寫代碼

POM.xml

<dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>
    </dependencies>

咱們這里采用SpringBoot 寫一個(gè)按需注入的方式開啟注解浙巫,編寫一個(gè)按需注入的注解

package com.hyh.log.annotation;

import com.hyh.log.config.HyhLogAutoConfiguration;
import org.springframework.context.annotation.Import;

import java.lang.annotation.*;

/**
 * 開啟日志注解
 *
 * @Author: heyuhua
 * @Date: 2021/1/28 15:32
 */
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Import({HyhLogAutoConfiguration.class})
public @interface EnableHyhLog {
}

自動(dòng)配置類

package com.hyh.log.config;

import com.hyh.log.aspect.LogAspect;
import com.hyh.log.service.LogService;
import com.hyh.log.service.impl.LogServiceImpl;
import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
 * Log AutoConfiguration
 *
 * @Author: heyuhua
 * @Date: 2021/1/28 16:02
 */
@ConditionalOnWebApplication
@Configuration(proxyBeanMethods = false)
public class HyhLogAutoConfiguration {

    @Bean
    public LogAspect logAspect() {
        return new LogAspect();
    }

    @Bean
    public LogService logService() {
        return new LogServiceImpl();
    }

}

自定義日志注解

package com.hyh.log.annotation;

import java.lang.annotation.*;

/**
 * 日志注解
 *
 * @Author: heyuhua
 * @Date: 2021/1/28 16:02
 */
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface HyhLog {

    /**
     * 描述
     *
     * @return
     */
    String value() default "";
}

復(fù)制代碼

日志切面代碼編寫

package com.hyh.log.aspect;

import com.hyh.log.annotation.HyhLog;
import com.hyh.log.service.LogService;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;

import javax.annotation.Resource;

/**
 * Log Aspect
 *
 * @Author: heyuhua
 * @Date: 2021/1/28 16:02
 */
@Aspect
public class LogAspect {

    /**
     * 日志服務(wù)
     */
    @Resource
    private LogService logService;

    /**
     * 環(huán)繞操作
     *
     * @param point
     * @param hyhLog
     * @return
     * @throws Throwable
     */
    @Around("@annotation(hyhLog)")
    public Object around(ProceedingJoinPoint point, HyhLog hyhLog) throws Throwable {
        String className = point.getTarget().getClass().getName();
        String methodName = point.getSignature().getName();
        logService.info("【類】:{},【方法】:{}", className, methodName);
        Object obj = point.proceed();
        // do something
        return obj;
    }

    /**
     * 前置操作
     *
     * @param point
     * @param hyhLog
     * @return
     * @throws Throwable
     */
    @Before("@annotation(hyhLog)")
    public void before(JoinPoint point, HyhLog hyhLog) throws Throwable {
        logService.info("執(zhí)行前置操作..." + hyhLog.value());
        // do something
    }

    /**
     * 后置操作
     *
     * @param point
     * @param hyhLog
     * @return
     * @throws Throwable
     */
    @After("@annotation(hyhLog)")
    public void after(JoinPoint point, HyhLog hyhLog) throws Throwable {
        logService.info("執(zhí)行后置操作..." + hyhLog.value());
        // do something
    }
}

復(fù)制代碼

日志服務(wù)接口

package com.hyh.log.service;

/**
 * Log 接口
 *
 * @Author: heyuhua
 * @Date: 2021/1/28 15:37
 */
public interface LogService {

    /**
     * info
     *
     * @param msg
     */
    void info(String msg);

    /**
     * info
     *
     * @param msg
     * @param o
     */
    void info(String msg, Object o);

    /**
     * info
     *
     * @param msg
     * @param throwable
     */
    void info(String msg, Throwable throwable);

    /**
     * info
     *
     * @param msg
     * @param o
     */
    void info(String msg, Object... o);

    /**
     * info
     *
     * @param msg
     * @param o
     * @param throwable
     */
    void info(String msg, Object o, Throwable throwable);

    /**
     * error
     *
     * @param msg
     */
    void error(String msg);

    /**
     * error
     *
     * @param msg
     * @param o
     */
    void error(String msg, Object o);

    /**
     * error
     *
     * @param msg
     * @param throwable
     */
    void error(String msg, Throwable throwable);

    /**
     * error
     *
     * @param msg
     * @param o
     * @param throwable
     */
    void error(String msg, Object o, Throwable throwable);

}

復(fù)制代碼

日志服務(wù)接口實(shí)現(xiàn)

package com.hyh.log.service.impl;

import com.hyh.log.service.LogService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Service;

/**
 * 日志服務(wù)接口實(shí)現(xiàn)
 *
 * @Author: heyuhua
 * @Date: 2021/1/28 17:04
 */
@Service
public class LogServiceImpl implements LogService {

    /**
     * 日志
     */
    private static final Logger LOGGER = LoggerFactory.getLogger(LogServiceImpl.class);

    @Override
    public void info(String msg) {
        LOGGER.info(msg);
    }

    @Override
    public void info(String msg, Object o) {
        LOGGER.info(msg, o);
    }

    @Override
    public void info(String msg, Throwable throwable) {
        LOGGER.info(msg, throwable);
    }

    @Override
    public void info(String msg, Object... o) {
        LOGGER.info(msg, o);
    }

    @Override
    public void info(String msg, Object o, Throwable throwable) {
        LOGGER.info(msg, o, throwable);
    }

    @Override
    public void error(String msg) {
        LOGGER.error(msg);
    }

    @Override
    public void error(String msg, Object o) {
        LOGGER.error(msg, o);
    }

    @Override
    public void error(String msg, Throwable throwable) {
        LOGGER.error(msg, throwable);
    }

    @Override
    public void error(String msg, Object o, Throwable throwable) {
        LOGGER.error(msg, o, throwable);
    }
}

復(fù)制代碼

三金蜀、單元測(cè)試

OK,不廢話的畴,直接上單元測(cè)試渊抄,先在SpringBoot項(xiàng)目中開啟使用我們的注解

AOP實(shí)戰(zhàn)篇 如何輕松實(shí)現(xiàn)日志功能,戳這

看紅框框選的我們寫的一個(gè)按需導(dǎo)入的注解丧裁,然后Controller層寫一個(gè)接口护桦,使用HyhLog注解

@HyhLog(value = "hello world test")
    @RequestMapping(value = "hello", method = RequestMethod.GET)
    public String hello() {
        return "hello";
    }

訪問hello接口來看下執(zhí)行結(jié)果

image.png

總結(jié)

AOP基于動(dòng)態(tài)代理可以實(shí)現(xiàn)很多功能,例如權(quán)限控制渣慕、日志記錄嘶炭、事務(wù)提交與回滾等一系列操作,通過無(wú)侵入式的方式實(shí)現(xiàn)業(yè)務(wù)解耦逊桦、功能強(qiáng)化,讓我們能用最優(yōu)雅的代碼實(shí)現(xiàn)最復(fù)雜的功能眨猎。

來源:https://www.tuicool.com/articles/FFJNNzf

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市强经,隨后出現(xiàn)的幾起案子睡陪,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,311評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件兰迫,死亡現(xiàn)場(chǎng)離奇詭異信殊,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)汁果,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門涡拘,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人据德,你說我怎么就攤上這事鳄乏。” “怎么了棘利?”我有些...
    開封第一講書人閱讀 152,671評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵橱野,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我善玫,道長(zhǎng)水援,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,252評(píng)論 1 279
  • 正文 為了忘掉前任茅郎,我火速辦了婚禮蜗元,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘只洒。我一直安慰自己许帐,他們只是感情好劳坑,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,253評(píng)論 5 371
  • 文/花漫 我一把揭開白布毕谴。 她就那樣靜靜地躺著,像睡著了一般距芬。 火紅的嫁衣襯著肌膚如雪涝开。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,031評(píng)論 1 285
  • 那天框仔,我揣著相機(jī)與錄音舀武,去河邊找鬼。 笑死离斩,一個(gè)胖子當(dāng)著我的面吹牛银舱,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播跛梗,決...
    沈念sama閱讀 38,340評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼寻馏,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了核偿?” 一聲冷哼從身側(cè)響起诚欠,我...
    開封第一講書人閱讀 36,973評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后轰绵,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體粉寞,經(jīng)...
    沈念sama閱讀 43,466評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,937評(píng)論 2 323
  • 正文 我和宋清朗相戀三年左腔,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了唧垦。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,039評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡液样,死狀恐怖业崖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情蓄愁,我是刑警寧澤双炕,帶...
    沈念sama閱讀 33,701評(píng)論 4 323
  • 正文 年R本政府宣布,位于F島的核電站撮抓,受9級(jí)特大地震影響妇斤,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜丹拯,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,254評(píng)論 3 307
  • 文/蒙蒙 一站超、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧乖酬,春花似錦死相、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至县昂,卻和暖如春肮柜,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背倒彰。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工审洞, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人待讳。 一個(gè)月前我還...
    沈念sama閱讀 45,497評(píng)論 2 354
  • 正文 我出身青樓芒澜,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親创淡。 傳聞我的和親對(duì)象是個(gè)殘疾皇子痴晦,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,786評(píng)論 2 345

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