首頁 > 軟體

Java實現AOP代理的三種方式詳解

2022-07-09 14:00:58

業務場景:首先你有了一個非常好的前輩無時無刻的在“教育”你。有這麼一天,它叫你將它寫好的一個方法進行改進測試,這時出現了功能迭代的情況。然後前輩好好“教育”你的說,不行改我的程式碼!改就腿打折!悲催的你有兩條路可走,拿出你10年跆拳道的功夫去火拼一波然後拍拍屁股瀟灑走人,要麼就是悲催的開始百度。。。這時你會發現,我擦怎麼把AOP代理這種事給忘了?【其實在我們工作中很少去手寫它,但是它又是很常見的在使用(控制檯紀錄檔)】

怎麼辦?打贏送手鐲,打輸睡大覺?

兄弟,信我的!

寫吧。。。

AOP是一種設計思想,是軟體設計領域中的面向切面程式設計,它是物件導向程式設計的一種補充和完善,它以通過預編譯方式和執行期動態代理方式實現在不修改原始碼的情況下給程式動態統一新增額外功能的一種技術。

利用AOP可以對業務邏輯的各個部分進行隔離,從而使得業務邏輯各部分之間的耦合度降低,提高程式的可重用性,同時提高了開發的效率

簡單說就是在不影響原有功能程式碼的情況下進行擴充套件,浸入少。

廢話不多說,開搞!

1、JDK實現

MyAop
package com.example.quasar.aop;
 
 
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
 
//繼承InvocationHandler介面實現
public class MyAop implements InvocationHandler {
    private Object object;
 
    public MyAop(Object object) {
        this.object = object;
    }
 
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //前置
        System.out.println("前置執行");
        Object invoke = method.invoke(this.object, args);
        //後置
        System.out.println("後置執行");
        return invoke;
    }
}

IndexService

package com.example.quasar.service;
 
 
public interface IndexService {
    public void run();
    public void run1();
}

IndexServiceImpl

package com.example.quasar.service.impl;
 
import com.example.quasar.service.IndexService;
 
public class IndexServiceImpl implements IndexService {
    @Override
    public void run() {
        System.out.println("執行了");
    }
 
    @Override
    public void run1() {
        System.out.println("執行了1");
    }
}

QuasarApplication

package com.example.quasar;
import com.example.quasar.aop.MyAop;
import com.example.quasar.service.IndexService;
import com.example.quasar.service.impl.IndexServiceImpl;
import org.springframework.boot.autoconfigure.SpringBootApplication;
 
import java.lang.reflect.Proxy;
 
@SpringBootApplication
public class QuasarApplication {
 
    public static void main(String[] args) {
        //實現類範例化
        IndexService indexService = new IndexServiceImpl();
        //將範例傳入aop
        MyAop myAop = new MyAop(indexService);
        //通過Proxy.newProxyInstance實現代理
        IndexService o = (IndexService) Proxy.newProxyInstance(IndexServiceImpl.class.getClassLoader(), new Class[]{IndexService.class}, myAop);
        //執行實現方法
        o.run();
        o.run1();
    }
 
}

執行結果

 2、CGLIB實現

如果專案中,已經使用了Spring,可以忽略匯入上述兩個包。因為在spring-core中已經整合了。否則需要引入該包,百度去找。

MyAop

package com.example.quasar.aop;
 
public class MyAop {
    public void befor() {
        System.out.println("befor...");
    }
    public void after() {
        System.out.println("after...");
    }
}

IndexService

package com.example.quasar.service;
 
 
public interface IndexService {
    public void run();
    public void run1();
}

IndexServiceImpl

package com.example.quasar.service.impl;
 
import com.example.quasar.service.IndexService;
 
public class IndexServiceImpl implements IndexService {
    @Override
    public void run() {
        System.out.println("執行了");
    }
 
    @Override
    public void run1() {
        System.out.println("執行了1");
    }
}

QuasarApplication

package com.example.quasar;
import com.example.quasar.aop.MyAop;
import com.example.quasar.service.IndexService;
import com.example.quasar.service.impl.IndexServiceImpl;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
 
import java.lang.reflect.Method;
 
@SpringBootApplication
public class QuasarApplication {
 
    public static void main(String[] args) {
        //實現類範例化
        IndexService indexService = new IndexServiceImpl();
        // 代理類 ,採用cglib,底層建立目標類的子類
        MyAop myAop = new MyAop();
        // 核心類
        Enhancer enhancer = new Enhancer();
        //確定父類別
        enhancer.setSuperclass(indexService.getClass());
        enhancer.setCallback(new MethodInterceptor() {
            @Override
            public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                //前置
                myAop.after();
                // 執行目標類的方法
                Object obj = method.invoke(indexService, args);
                //後置
                myAop.befor();
                return obj;
            }
        });
        // 3.4 建立代理
        IndexService proxService = (IndexService) enhancer.create();
        proxService.run();
        proxService.run1();
    }
 
}

執行結果

 3、boot註解實現【注意只對bean有效】

MyAop

package com.v1.aop;
 
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.Before;
import org.springframework.stereotype.Component;
 
@Aspect
@Component
public class MyAop {
    //*號作用:com.v1.service.impl下的所有類和所有方法
    @Before(value = "within(com.v1.service.impl.*)")
    public void before(JoinPoint joinPoint) {
        System.out.println("before開始執行查詢.......");
        System.out.println("正在執行的目標類是: " + joinPoint.getTarget());
        System.out.println("正在執行的目標方法是: " + joinPoint.getSignature().getName());
    }
    //*號作用:com.v1.controller下的所有類和所有方法
    @Around(value = "execution(* com.v1.controller.*.*(..))")
    public Object aroud(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        System.out.println("aroud環繞通知開始.......");
        System.out.println("執行的目標類 = " + proceedingJoinPoint.getTarget());
        System.out.println("執行的目標方法 = " + proceedingJoinPoint.getSignature().getName());
        // 必須方法目標方法
        Object proceed = proceedingJoinPoint.proceed();
        System.out.println("aroud環繞通知結束.......");
        // 將目標方法的返回值進行返回,否則呼叫目標方法的方法無法獲取到返回值
        return proceed;
    }
 
}

非常簡單,直接就根據註解進行代理了!

註解引數可以具體在咱們csdn平臺去查一下,有很多文章講的很細的!

執行結果

發個請求跑一下

以上就是Java實現AOP代理的三種方式詳解的詳細內容,更多關於Java AOP代理的資料請關注it145.com其它相關文章!


IT145.com E-mail:sddin#qq.com