zoukankan      html  css  js  c++  java
  • Java中两种动态代理的实现

    本文介绍了java中两种动态代理的实现方法,Spring的动态代理也是基于这两种方法的。直接附上源码:

    1、JDK实现

    使用JDK实现动态代理必须使用接口

    • 接口Work.java
    public interface Work {
        public void work();
    }
    • 实现类WorkImpl.java
    public class WorkImpl implements Work {
    
        @Override
        public void work() {
            System.out.println("我在工作");
        }
    
    }
    • 一个添加日志的代理类LogHandler.java,这里用到了log4j。
    public class LogHandler<T> implements InvocationHandler{
        
        public T target;
        
        private static Logger logger = Logger.getLogger(LogHandler.class);
        
        public LogHandler(T target){
            this.target = target;
        }
        
        @SuppressWarnings("unchecked")
        public T getInstance(){
            return (T) Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
        }
        
        @Override
        public Object invoke(Object proxy, Method method, Object[] args)
                throws Throwable {
            logger.warn("start");
            Object result = method.invoke(target, args);
            logger.warn("end");
            return result;
        }
    
    }
    • 最后测试类Test.java
    public class Test {
        public static void main(String[] args) {
            Work w = new WorkImpl();
            w.work();
            Work logWork =  new LogHandler<Work>(w).getInstance();
            logWork.work();
        }
    }

    2、CGLIB实现

    CGLIB实现动态代理是对JDK动态代理的一种补充,可以不实现接口就能动态代理。

    首先需要引入CGLIB的相关jar包(cglib-nodep-2.2.2.jar)

    • 业务相关类WorkImpl.java
    public class WorkImpl   {
    
        public void work() {
            System.out.println("我在工作");
        }
    
    }
    • CGLIB动态代理类LogHandler.java,同样是添加日志信息
    public class LogHandler<T> implements MethodInterceptor{
        
        public T target;
        
        private static Logger logger = Logger.getLogger(LogHandler.class);
        
        public LogHandler(T target){
            this.target = target;
        }
        
        @SuppressWarnings("unchecked")
        public T getInstance(){
            //使用CGLIB生成子类,并委托代理对象代理
            return (T) Enhancer.create(target.getClass(), this);
        }
        
        @Override
        public Object intercept(Object obj, Method method, Object[] args,
                MethodProxy proxy) throws Throwable {
            logger.warn("start!");
            Object result=proxy.invokeSuper(obj, args);
            logger.warn("end!");
            return result;
        }
        
    }
    • 测试类Test.java
    public class Test {
        public static void main(String[] args) {
            WorkImpl w = new WorkImpl();
            w.work();
            WorkImpl logWork = new LogHandler<WorkImpl>(w).getInstance();
            logWork.work();
        }
    }
  • 相关阅读:
    利用.net Core 对程序集中的类 进行统一依赖注入
    接口中定义异步的方法
    在efcore 中创建类 通过实现IEntityTypeConfiguration<T>接口 实现实体类的伙伴类 实现FluentApi
    在vs2017 版本15.7.6中不支持2.1.0以上版本的net core sdk
    在.net core不同的版本中 webabi引用的包不同
    SQL语句中使用Group by
    DDD实战12 值对象不创建表,而是直接作为实体中的字段
    src与href的区别。
    cookies,sessionStorage 和 localStorage 的区别
    javascript阻止事件冒泡和浏览器的默认行为
  • 原文地址:https://www.cnblogs.com/sunxueqiang0329/p/3718383.html
Copyright © 2011-2022 走看看