真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

如何解決Springaop失效問題

這篇文章主要為大家展示了如何解決Spring aop失效問題,內(nèi)容簡而易懂,希望大家可以學(xué)習(xí)一下,學(xué)習(xí)完之后肯定會有收獲的,下面讓小編帶大家一起來看看吧。

創(chuàng)新互聯(lián)公司長期為上1000+客戶提供的網(wǎng)站建設(shè)服務(wù),團(tuán)隊(duì)從業(yè)經(jīng)驗(yàn)10年,關(guān)注不同地域、不同群體,并針對不同對象提供差異化的產(chǎn)品和服務(wù);打造開放共贏平臺,與合作伙伴共同營造健康的互聯(lián)網(wǎng)生態(tài)環(huán)境。為馬龍企業(yè)提供專業(yè)的成都網(wǎng)站制作、做網(wǎng)站,馬龍網(wǎng)站改版等技術(shù)服務(wù)。擁有十載豐富建站經(jīng)驗(yàn)和眾多成功案例,為您定制開發(fā)。

先看下這個(gè)問題的背景:假設(shè)有一個(gè)spring應(yīng)用,開發(fā)人員希望自定義一個(gè)注解@Log,可以加到指定的方法上,實(shí)現(xiàn)自動記錄日志(入?yún)?、出參、響?yīng)耗時(shí)這些)

package com.cnblogs.yjmyzz.springbootdemo.aspect;
 
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
 
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Log {
 
}

然后再寫一個(gè)Aspect來解析這個(gè)注解,對打了Log注解的方法進(jìn)行增強(qiáng)處理 

package com.cnblogs.yjmyzz.springbootdemo.aspect;
 
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
 
import java.lang.reflect.Method;
 
@Component
@Aspect
public class LogAspect {
 
  @Pointcut("execution (* com.cnblogs.yjmyzz.springbootdemo.service..*.*(..))")
  public void logPointcut() {
 
  }
 
  @Around("logPointcut()")
  public void around(JoinPoint point) {
    String methodName = point.getSignature().getName();
    Object[] args = point.getArgs();
    Class<?>[] argTypes = new Class[point.getArgs().length];
    for (int i = 0; i < args.length; i++) {
      argTypes[i] = args[i].getClass();
    }
    Method method = null;
    try {
      method = point.getTarget().getClass().getMethod(methodName, argTypes);
    } catch (Exception e) {
      e.printStackTrace();
    }
    //獲取方法上的注解
    Log log = method.getAnnotation(Log.class);
    if (log != null) {
      //演示方法執(zhí)行前,記錄一行日志
      System.out.println("before:" + methodName);
    }
    try {
      //執(zhí)行方法
      ((ProceedingJoinPoint) point).proceed();
    } catch (Throwable throwable) {
      throwable.printStackTrace();
    } finally {
      if (log != null) {
        //演示方法執(zhí)行后,記錄一行日志
        System.out.println("after:" + methodName);
      }
    }
  }
}

寫一個(gè)測試Service類:

package com.cnblogs.yjmyzz.springbootdemo.service;
 
import com.cnblogs.yjmyzz.springbootdemo.aspect.Log;
import org.springframework.stereotype.Component;
 
@Component
public class HelloService {
   
  @Log
  public void sayHi(String msg) {
    System.out.println("\tsayHi:" + msg);
  }
 
  public void anotherSayHi(String msg) {
    this.sayHi(msg);
  }
 
}

最后來跑一把:

package com.cnblogs.yjmyzz.springbootdemo;
 
import com.cnblogs.yjmyzz.springbootdemo.service.HelloService;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
 
/**
 * @author 菩提樹下的楊過
 */
@ComponentScan("com.cnblogs.yjmyzz")
@Configuration
@EnableAspectJAutoProxy
public class SampleApplication {
 
  public static void main(String[] args) {
    AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(SampleApplication.class);
    HelloService helloService = context.getBean(HelloService.class);
    helloService.sayHi("hi-1");
    System.out.println("\n");
    helloService.anotherSayHi("hi-2");
  }
}

輸出如下:

如何解決Spring aop失效問題

顯然HelloService中的anotherSayHi方法,并未被aop增強(qiáng)。 原因其實(shí)很簡單,了解AOP原理的同學(xué)想必都知道,AOP的實(shí)現(xiàn)有二類,如果是基于接口的,會采用動態(tài)代理,生成一個(gè)代理類,如果是基于類的,會采用CGLib生成子類,然后在子類中擴(kuò)展父類中的方法。

如何解決Spring aop失效問題

本文中HelloService并不是一個(gè)接口,所以從上圖的斷點(diǎn)中可以看出,當(dāng)Spring運(yùn)行時(shí),HelloService被增加為...EnhancerBySpringCGLib...。但是當(dāng)調(diào)用到anotherSayHi時(shí)

如何解決Spring aop失效問題

方法的調(diào)用方,其實(shí)是原始的HelloSerfvice實(shí)例,即:是未經(jīng)過Spring AOP增強(qiáng)的對象實(shí)例。所以解決問題的思路就有了,想辦法用增強(qiáng)后的HelloService實(shí)例來調(diào)用!

方法一:用Autowired 注入自身的實(shí)例

如何解決Spring aop失效問題

這個(gè)方法,第一眼看上去感覺有些怪,自己注入自己,感覺有點(diǎn)象遞歸/死循環(huán)的搞法,但確實(shí)可以work,Spring在解決循環(huán)依賴上有自己的處理方式,避免了死循環(huán)。

方法二:從Spring上下文獲取增強(qiáng)后的實(shí)例引用

如何解決Spring aop失效問題

原理與方法一其實(shí)類似,不多解釋。

方法三: 利用AopContext

如何解決Spring aop失效問題

不過這個(gè)方法要注意的是,主類入口上,必須加上exporseProxy=true,參考下圖:

如何解決Spring aop失效問題

最后來驗(yàn)證下這3種方法是否生效:

如何解決Spring aop失效問題

從運(yùn)行結(jié)果上看,3種方法都可以解決這個(gè)問題?!?/p>

以上就是關(guān)于如何解決Spring aop失效問題的內(nèi)容,如果你們有學(xué)習(xí)到知識或者技能,可以把它分享出去讓更多的人看到。


分享題目:如何解決Springaop失效問題
分享網(wǎng)址:http://weahome.cn/article/iiedjs.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部