Java随笔-动态代理

代理

代理是一种基本的设计模式,为目标对象提供另外的访问方式,通过代理访问目标对象可以扩展目标对象的功能,增加额外的功能,也就是不要修改别人的代码,但是可以扩展。
代理分为三种:静态代理、动态代理、CGLIB代理.

静态代理

静态代理就是在代码中把需要代理的类和代理的方法确定好了。

  1. 创建公共接口。
public interface Worker {
    void work();
}
  1. 创建实现类,确定具体的工作。
public class Programmer implements Worker{
    @Override
    public void work() {
        System.out.println("程序员正在使劲搬砖中。。。");
    }
}
  1. 创建代理类。
public class OutSouring implements Worker{
    private Worker worker;

    public OutSouring(Worker worker) {
        this.worker = worker;
    }

    @Override
    public void work() {
        System.out.println("外包要接项目了");
        worker.work();
    }
}

代理类同样需要实现公共接口,持有相关对象,不然没有办法让真正的干活的人干活。通常持有的是具体的实现类对象。

public class OutSouring implements Worker{
    private Programmer programmer;
    public OutSouring(Programmer programmer) {
        this.programmer = programmer;
    }

    @Override
    public void work() {
        System.out.println("外包要接项目了");
        programmer.work();
    }
}

  1. 代理类给实现类派活。
    public static void main(String[] args) {
        Programmer programmer = new Programmer();
        OutSouring outSouring = new OutSouring(programmer);
        outSouring.work();
    }

结果:

外包要接项目了
程序员正在使劲搬砖中。。。

其实就是干活时,自己指挥别人干活。

动态代理

动态代理就是根据代码的指示动态代理,并没有指明确需要代理的工作。动态代理的好处是可以对代理类的所有方法进行统一管理,比较方便,不需要对代理的方法每一个都写一遍;不好的地方在于动态代理基于反射,可能稍稍稍慢一点吧。

  1. 创建公共接口,同上。
  2. 创建实现类,确定具体的工作,同上。
  3. 创建ProgrammerInvocationHandler类,实现InvocationHandler接口,持有被代理对象,实现invoke方法。
public class ProgrammerInvocationHandler implements InvocationHandler {
    private Object proxied;

    public ProgrammerInvocationHandler(Object proxied) {
        this.proxied = proxied;
    }
    
    /**
     * 调用方法
     * @param proxy     代理对象
     * @param method    调用的方法
     * @param args      方法传参
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("准备开始干活了");
        Object result = method.invoke(proxied, args);
        return result;
    }
}

invoke就是在持有被代理对象后,通过反射执行被代理对象中的所有方法,当然也可以做自己的工作。当然这里指定了具体被代理的类型,也可以使用泛型。

public class ProgrammerInvocationHandler<T> implements InvocationHandler {
    private T proxied;

    public ProgrammerInvocationHandler(T proxied) {
        this.proxied = proxied;
    }

    /**
     * 调用方法
     * @param proxy     代理对象
     * @param method    调用的方法
     * @param args      方法传参
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("准备开始干活了");
        Object result = method.invoke(proxied, args);
        return result;
    }
}
  1. 创建代理对象,执行代理方法。
        // 创建被代理对象
        Programmer programmer = new Programmer();
        // 创建InvocationHandler对象
        ProgrammerInvocationHandler programmerInvocationHandler = new ProgrammerInvocationHandler(programmer);
        // 创建代理对象,代理对象会自动执行相应的方法
        Worker proxy = (Worker) Proxy.newProxyInstance(Worker.class.getClassLoader(), new Class[]{Worker.class}, programmerInvocationHandler);
        proxy.work();

通过调用静态方法Proxy.newProxyInstance()就可以创建动态代理,需要类加载器,被实现的接口,相关方法处理入口。传入programmerInvocationHandler,其实就是指定了将要被调用的方法。

    @CallerSensitive
    public static Object newProxyInstance(ClassLoader loader,
                                          Class<?>[] interfaces,
                                          InvocationHandler h) {
        Objects.requireNonNull(h);

        final Class<?> caller = System.getSecurityManager() == null
                                    ? null
                                    : Reflection.getCallerClass();

        /*
         * Look up or generate the designated proxy class and its constructor.
         */
        Constructor<?> cons = getProxyConstructor(caller, loader, interfaces);

        return newProxyInstance(caller, cons, h);
    }
    ...
    private static Constructor<?> getProxyConstructor(Class<?> caller,
                                                      ClassLoader loader,
                                                      Class<?>... interfaces)
    {
        // optimization for single interface
        if (interfaces.length == 1) {
            Class<?> intf = interfaces[0];
            if (caller != null) {
                checkProxyAccess(caller, loader, intf);
            }
            return proxyCache.sub(intf).computeIfAbsent(
                loader,
                (ld, clv) -> new ProxyBuilder(ld, clv.key()).build()
            );
        } else {
            // interfaces cloned
            final Class<?>[] intfsArray = interfaces.clone();
            if (caller != null) {
                checkProxyAccess(caller, loader, intfsArray);
            }
            final List<Class<?>> intfs = Arrays.asList(intfsArray);
            return proxyCache.sub(intfs).computeIfAbsent(
                loader,
                (ld, clv) -> new ProxyBuilder(ld, clv.key()).build()
            );
        }
    

Proxy.newProxyInstance()实际上就是通过反射获取代理对象构造器,然后用构造器创建具体的对象。
结果:

准备开始干活了
程序员正在使劲搬砖中。。。

动态代理在代码运行时才知道具体的实现,在面向切面编程(AOP)中有使用。

CGLIB

如果被代理的类没有实现接口,那么动态代理就没有办法使用,这个时候就需要使用CGLIB代理。Cglib代理也叫作子类代理,是通过在内存中构建一个子类,并在子类中采用方法拦截的技术拦截所有父类方法的调用,然后加入自己需要的操作,这其实就是面向切面编程的思想。CGLIB不能代理final 类,因为使用的是继承。使用之前需要添加依赖。

  1. 创建被代理对象。
public class ProxyObject {

    public void doWork(){
        System.out.println("干活是不可能的,除非。。。。");
    }
    
}
  1. 创建代理工厂。
public class ProxyFactory{

	public ProxyFactory() {
	}
	public static <T> T getProxy(T proxied) {
		// 工具类
		Enhancer enhancer = new Enhancer();
		// 设置父类
		enhancer.setSuperclass(proxied.getClass());
		// 设置回调
		enhancer.setCallback(new MethodInterceptor() {
			
			@Override
			public Object intercept(Object obj, Method method, Object[] args, MethodProxy arg3) throws Throwable {
				System.out.println("准备干活了。。。。");
				Object object = method.invoke(target, args);
				return object;
			}
		});
		
		@SuppressWarnings("unchecked")
		T proxy = (T) enhancer.create();
		return proxy;
	}
}
  1. 开始代理。
		ProxyObject proxied= new ProxyObject ();
		ProxyObject proxy = (ProxyObject)ProxyFactory.getProxy(proxied);
		// 干活
		proxy.doWork();
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值