相同類中方法間調用時日誌Aop失效處理

本篇分享的內容是在相同類中方法間調用時Aop失效處理方案,該問題我看有不少文章描述了,不過大可能是從事務角度分享的,本篇打算從日誌aop方面分享(固然都是aop,失效和處理方案都是同樣),如下都是基於springboot演示;spring

  • 快速定義個日誌Appender
  • 快速定義個攔截器和日誌註解(aop)
  • 模擬相同類中方法間調用時aop失效
  • Aop失效處理方案(就兩種足夠了)

快速定義個日誌Appender

日誌我仍是喜歡log4j,大部分朋友也一樣吧,這裏lombok與log4j結合來完成咱們的日誌,以下maven包(最新mvn仍是建議去官網找):api

<dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>

        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-api</artifactId>
            <version>2.0.0-alpha0</version>
        </dependency>
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-log4j12</artifactId>
            <version>2.0.0-alpha0</version>
        </dependency>

先繼承log4j的AppenderSkeleton重寫下append方法,簡單記錄下就行,以下:springboot

public class MyLogAppend extends AppenderSkeleton {
    private String author;

    public void setAuthor(String author) {
        this.author = author;
    }

    @Override
    protected void append(LoggingEvent loggingEvent) {
        System.out.println(
                JsonUtil.formatMsg("date -- {},level -- {},message -- {}",
                        LocalDate.now(),
                        loggingEvent.getLevel(),
                        loggingEvent.getMessage()));
    }

    @Override
    public void activateOptions() {
        super.activateOptions();
        System.out.println("author:" + this.author);
    }

    @Override
    public void close() {
        this.closed = true;
    }

    @Override
    public boolean requiresLayout() {
        return false;
    }
}

而後項目根目錄增長log4j.properties配置文件,配置內容定義info級別,就此完成了log4j自定義記錄日誌了:app

log4j.rootLogger=info,MyLogAppend
log4j.appender.MyLogAppend=com.sm.component.log.MyLogAppend
log4j.appender.MyLogAppend.author=shenniu003

快速定義個攔截器和日誌註解(aop)

一般同類中不一樣方法調用是常事,能夠直接用this.xx();有時有這樣需求,須要各個調用方法時候的參數記錄下來,所以咱們須要個攔截器,再增長個自定義註解方便使用:maven

@Aspect
@Component
@Slf4j
public class MyLogInterceptor {

    private final String pointcut = "@annotation(com.sm.component.ServiceLog)";

    @Pointcut(pointcut)
    public void log() {
    }

    @Before(value = "log()")
    void before(JoinPoint joinPoint) {
        Signature signature = joinPoint.getSignature();
        log.info(
                JsonUtil.formatMsg("method:{},params:{}",
                        signature.toLongString(),
                        joinPoint.getArgs()));
    }
}
@Documented
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ServiceLog {
}

攔截器攔截帶有@ServiceLog註解的方法,而後記錄請求參數和方法名ide

模擬相同類中方法間調用時aop失效

利用上面完成的日誌註解,這裏在OrderService類中用getOrderDetail方法去調用getOrderLog方法,他兩都標記日誌註解便於記錄參很多天志;同時getOrderDetail方法也調用另一個UserService類中的getNickName方法,便於比較:ui

@Service
public class OrderService {

    @Autowired
    UserService userService;

    @ServiceLog
    public String getOrderDetail(String orderNum) {
        String des = "訂單號【" + orderNum + "】月餅一盒";

        userService.getNickName(orderNum);

        this.getOrderLog(orderNum + "11111");

        return des;
    }

    @ServiceLog
    public List<String> getOrderLog(String orderNum) {
        List<String> logs = new ArrayList<>();
        IntStream.range(0, 5).forEach(b -> {
            logs.add("用戶" + b + "購買成功");
        });
        return logs;
    }
}
@Service
public class UserService {
    @ServiceLog
    public String getNickName(String userId) {
        return "神牛" + userId;
    }
}

方法調用重點截圖:
image
而後運行程序,接口觸發調用getOrderDetail方法,如下攔截器中記錄的日誌信息:
image
可以看出攔截器只記錄到了getOrderDetail和getNickName方法的日誌,所以能夠確定getOrderLog根本沒有走攔截器,儘管在方法上加了日誌@ServiceLog註解也沒用。this

Aop失效處理方案(就兩種足夠了)

就上面相同類中方法間調用攔截器(aop)沒起做用,咱們有以下經常使用兩種方式處理方案;spa

  1. 用@Autowired或Resource引入自身依賴
  2. 開啓暴露代理類,AopContext.currentProxy()方式獲取代理類

第一種:主要使用註解方法引入自身代理依賴,不要使用構造的方式會有循環依賴問題,如下使用方式:
image
第二種:經過暴露代理類方式,實際原理是把代理類添加到當前請求的ThreadLocal裏面,而後在使用時從ThreadLocal中獲取代理類,再調用對應的方法,開啓方式須要:代理

@EnableAspectJAutoProxy(exposeProxy = true)

而後方法中以下使用便可:
image
最後來看下使用這兩種方式正常走攔截器效果:
image

無論是日誌攔截器或事務,他們都是aop的方式,底層原理走的代理方式,只有使用代理類纔會正常執行攔截器,而this.xxx()使用的是自身實例對象,所以會出現上面失效的狀況。

 

原文連接

本文爲雲棲社區原創內容,未經容許不得轉載。

相關文章
相關標籤/搜索