• Spring-Aop的两种代理方式


    Spring-Aop两种代理方式:

    1、JDK动态代理:用于目标类实现了接口;

    2、Cglib动态代理:用于目标类没有实现接口;

    spring会依据目标类是否实现接口来选择使用哪种代理方式(目标类:相当于需要被增强的类);

    模拟场景:目标类:步兵类,需要上战场打仗(手提精心打造的木棍)

    通知类(增强类):高级商店类,只要有士兵来就免费提供(精心打造的暴风大剑)

    代理工厂类:需要一个士兵来使用商店的东西(这样才能把士兵与商店联系起来,士兵与商店本来是单独存在的相互不知道)

    JDK动态代理实现:

    创建maven项目引入spring-aop依赖,使用的是5.1版本

    1、目标类需要实现接口,所以先创建一个接口:SoldierInterface(士兵类的接口)

    package top.free;
    
    /**
     * 士兵接口
     */
    public interface SoldierInterface {
        public void begin();
    }

    2、创建目标类(步兵类去实现接口),也可以有其他的类(弓箭兵。。。)只要实现了士兵类,都是士兵。

    package top.free;
    
    /*
     *步兵类
     */
    public class Soldier implements SoldierInterface {
    
        @Override
        public void begin() {
            System.out.println("弓箭士兵开始战斗");
        }
    }

    3、通知(也叫增强):商店类

    package top.free.advice;
    
    public class Store {
        public void Before(){
            System.out.println("为士兵换上暴风大剑。。。");
        }
        public void After(){
            System.out.println("士兵胜利归来。。。");
        }
    }

    4、创建代理工厂:需要传递过来一个士兵(不管是步兵还是弓箭兵),返回一个被商店加强的士兵

    package top.free.proxy;
    
    
    import top.free.SoldierInterface;
    import top.free.advice.Store;
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    public class MyProxyFactory {   //传递的soldier就是需要被增强的士兵类, public SoldierInterface MyProxyFactory(SoldierInterface soldier){ //通知,此时的商店类,在工厂中new出的实列 Store store = new Store(); SoldierInterface proxysoldier = (SoldierInterface) Proxy.newProxyInstance( MyProxyFactory.class.getClassLoader(),//第一个参数:使用当前类的类加载器 soldier.getClass().getInterfaces(),//第二个参数:传递的soldier士兵对象的接口 new InvocationHandler() {//第三个参数,匿名内部类形式 @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //前置调用商店的配置 store.Before(); //调用目标方法,soldier就是传递的士兵对象,args参数默认无参 Object invoke = method.invoke(soldier, args); //后置,士兵执行完方法后调用 store.After(); return invoke; } } ); return proxysoldier; } }

     5、测试运行:

    package top.free;
    
    import top.free.proxy.MyProxyFactory;
    
    public class App 
    {
        public static void main(String[] args) {
            //创建工厂
            MyProxyFactory myProxyFactory = new MyProxyFactory();
            //创建目标类(士兵对象),具体的实现是步兵类
            SoldierInterface soldier = new Soldier();
            //将创建的士兵传递给工厂,工厂中有商店帮忙打造士兵,用新士兵接收
            SoldierInterface newSoldier = myProxyFactory.MyProxyFactory(soldier);
            //开始
            newSoldier.begin();
        }
    }

     

    Cglib动态代理实现:

     写Jdk代理时已引入spring相关依赖,此时maven项目中已有spring-core包,Spring5.1版本中,spring-core包中有关于Cglib的类,不需要引入cglib依赖

    1、Cglib代理使用的是没有接口的目标类,所以直接创建弓箭兵:

    package top.free;
    
    /**
     * Cglib目标类,没有接口,弓箭兵类
     */
    public class BowsSoldier {
        public void begin(){
            System.out.println("开始射杀敌人。。。");
        }
    }

    2、通知(商店类):

    package top.free.advice;
    
    public class Store {
    
        public void Before(){
            System.out.println("为士兵换上霸王弓。。。");
        }
        public void After(){
            System.out.println("士兵胜利归来。。。");
        }
    }

    3、创建代理工厂:需要一个弓箭兵,返回一个装备好的弓箭兵

    package top.free.proxy;
    
    import org.springframework.cglib.proxy.Enhancer;
    import org.springframework.cglib.proxy.MethodInterceptor;
    import org.springframework.cglib.proxy.MethodProxy;
    import top.free.BowsSoldier;
    import top.free.advice.Store;
    import java.lang.reflect.Method;
    
    public class MyCglibProxyFactory {
        public BowsSoldier MyCglibProxyFactory(BowsSoldier solider){
            //创建通知类
            Store store = new Store();
            //创建Enhance对象
            Enhancer enhancer = new Enhancer();
            //设置弓箭兵类对象
            enhancer.setSuperclass(solider.getClass());
            //设置回调方法
            enhancer.setCallback(new MethodInterceptor() {
                @Override
                public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                    //放在方法执行前,就是前置增强,装备弓箭
                    store.Before();
                    //执行方法,solider就是传递过来的弓箭兵对象
                    Object invoke = method.invoke(solider);
                    //后置增强
                    store.After();
                    return invoke;
                }
            });
            //创建装备好的士兵
            BowsSoldier target1= (BowsSoldier) enhancer.create();
            return target1;
        } 
    }

     4、测试:

    package top.free;
    
    import top.free.proxy.MyCglibProxyFactory;
    import top.free.proxy.MyProxyFactory;
    
    public class App 
    {
        public static void main(String[] args) {//创建代理工厂
            MyCglibProxyFactory myCglibProxyFactory = new MyCglibProxyFactory();
            //创建一个普通的弓箭兵
            BowsSoldier soldier = new BowsSoldier();
            //传递弓箭兵到工厂,工厂装备好后返回,接收一个装备精良的弓箭兵
            BowsSoldier target = myCglibProxyFactory.MyCglibProxyFactory(soldier);
            //装备精良的弓箭兵开始战斗
            target.begin();
        }
    }

     

  • 相关阅读:
    如何对抗硬件断点--- 调试寄存器
    学会破解要十招
    Ollydbg 中断方法浅探
    脱壳经验(二)--如何优化
    jquery 选择器,模糊匹配
    CSS实现内容超过长度后以省略号显示
    微信内置浏览器浏览H5页面弹出的键盘遮盖文本框的解决办法
    Exif.js获取图片的详细信息(苹果手机移动端上传图片旋转90度)
    jquery中的$("#id")与document.getElementById("id")的区别
    CentOS7 安装lua环境(我是在mysql读写分离用的)
  • 原文地址:https://www.cnblogs.com/YpfBolg/p/10890876.html
Copyright © 2020-2023  润新知