Spring动态代理

代理模式之前也有写过,不过只有静态实现
https://blog.csdn.net/qq_33321609/article/details/87859654
代理模式:给某一个对象提供一个代理或占位符,并由代理对象来控制对原对象的访问。
如果是静态代理,我没要实现一个功能,就要写两遍代码,实现中写一遍,代理器中写一遍
所以动态代理应运而生,动态代理最重要的是实现InvocationHandler接口
话不多说,直接上代码

接口

public interface UserManager {
    @EsbMethod2(name = "addUser")
    public void addUser();

    @EsbMethod(name = "deleteUser")
    public void deleteUser();
}

接口的实现

public class UserManagerImpl implements UserManager {
    @Override
    public void addUser() {
        System.out.println("添加用户");
    }

    @Override
    public void deleteUser() {
        System.out.println("删除用户");
    }
}

代理

public class UserHandler implements InvocationHandler {
    // 目标对象
    private Object targetObject;

    private List<Aspect> aspects = new ArrayList<>();

    public void addAspect(Aspect aspect) {
        aspects.add(aspect);
    }

    //绑定关系,也就是关联到哪个接口(与具体的实现类绑定)的哪些方法将被调用时,执行invoke方法。
    public Object newProxyInstance(Object targetObject) {
        this.targetObject = targetObject;
        //该方法用于为指定类装载器、一组接口及调用处理器生成动态代理类实例
        //第一个参数指定产生代理对象的类加载器,需要将其指定为和目标对象同一个类加载器
        //第二个参数要实现和目标对象一样的接口,所以只需要拿到目标对象的实现接口
        //第三个参数表明这些被拦截的方法在被拦截时需要执行哪个InvocationHandler的invoke方法
        //根据传入的目标返回一个代理对象
        return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),
                targetObject.getClass().getInterfaces(), this);
    }

    @Override
    //关联的这个实现类的方法被调用时将被执行
    /*InvocationHandler接口的方法,proxy表示代理,method表示原对象被调用的方法,args表示方法的参数*/
    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        System.out.println("start-1->>");
        if (args != null) {
            for (int i = 0; i < args.length; i++) {
                System.out.println(args[i]);
            }
        }
        Object ret = null;
        try {
            /*原对象方法调用前处理日志信息*/
            System.out.println("satrt-->>");
            //调用目标方法
            if (method.isAnnotationPresent(EsbMethod.class)) {
                EsbMethod m = method.getAnnotation(EsbMethod.class);
                System.out.println(m.name());
            }
            for (Aspect aspect : aspects) {
                if (method.isAnnotationPresent(aspect.methodClass())) {
                    aspect.before();
                }
            }
            ret = method.invoke(targetObject, args);
            for (Aspect aspect : aspects) {
                if (method.isAnnotationPresent(aspect.methodClass())) {
                    aspect.after();
                }
            }
            /*原对象方法调用后处理日志信息*/
            System.out.println("success-->>");
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println("error-->>");
            throw e;
        }
        return ret;
    }

}

Test

public class Test {
    public static void main(String[] args) {
        MyAspect myAspect = new MyAspect();
        MyAspect2 myAspect2 = new MyAspect2();
        UserHandler userHandler = new UserHandler();
        userHandler.addAspect(myAspect);
        userHandler.addAspect(myAspect2);
        UserManager userManager = (UserManager) userHandler.newProxyInstance(new UserManagerImpl());
        userManager.addUser();
        userManager.deleteUser();
    }
}

是不是感觉有些没用的代码
是的,我在试试使用动态代理模式加入拦截器,不是所有的都拦截,而是在加了特定注解的方法中加入拦截,拦截很简单,只是打印一行文字,如果需要复杂,可以加很多东西,至于真正的拦截器如何实现,我不是很清楚

拦截所需的枚举

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface EsbMethod {
    String name() default "";
}
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface EsbMethod2 {
    String name() default "";
}

拦截接口

public interface Aspect {

    public Class methodClass();

    public void before();

    public void after();
}

拦截实现类

public class MyAspect implements Aspect {

    @Override
    public Class methodClass() {
        return EsbMethod.class;
    }

    @Override
    public void before() {
        System.out.println("拦截 MyAspect before");
    }

    @Override
    public void after() {
        System.out.println("拦截 MyAspect after");
    }
}
public class MyAspect2 implements Aspect {

    @Override
    public Class methodClass() {
        return EsbMethod2.class;
    }

    @Override
    public void before() {
        System.out.println("拦截 MyAspect2 before");
    }

    @Override
    public void after() {
        System.out.println("拦截 MyAspect2 after");
    }
}

输出

start-1->>
satrt-->>
MyAspect2 before
添加用户
MyAspect2 after
success-->>
start-1->>
satrt-->>
deleteUser
MyAspect before
删除用户
MyAspect after
success-->>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值