设计模式之代理模式学习

1 代理模式概念介绍

代理模式定义:为其他对象提供一种代理以控制对这个对象的访问。代理对象起到中介作用,可去掉功能服务或增加额外的服务。

1.1 常见代理模式

(1)远程代理:为不同地理的对象提供局域网代表对象。

(2)虚拟代理:根据需要将资源消耗很大的对象进行延迟真正需要的时候进行创建。

(3)保护代理:控制对一个对象的访问权限控制。

(4)智能引用代理:提供对目标对象额外服务。

2 商用代理模式原理

2.1 静态代理概念及实现

静态代理:代理和被代理对象在代理之前是确定的。他们都实现相同的接口 或者继承相同的抽象类。

2.2 动态代理

Java动态代理类位于java.lang.reflect包下,一般主要涉及到以下两个类:

(1)Interface InvocationHandler:该接口中仅定义了一个方法

public object invoke(Object obj,Method method,Object[] args)

在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,args为该方法的参数数组。这个抽象方法在代理类中动态实现。

(2)Proxy:该类即为动态代理类

static Object newProxyInstance(ClassLoader loader,Class []interfaces,InvocationHandler h):返回代理类的一个实例,返回后的代理类可以当做被代理类使用(可使用被代理类的在接口中声明过的方法)

JDK动态代理(Dynamic Proxy)是这样一种class:

它是在运行时产生的class

该class需要实现一组interface

使用动态代理类时,必须实现InvocationHandler接口。

2.3 动态代理类实现步骤

(1)创建一个接口InvocationHandler的类,它必须实现invoke方法。
(2)创建被代理的类以及接口

(3)调用Proxy的静态方法,创建一个代理类

(4)通过代理类调用方法

newProxyInstance(ClassLoader loader,Class[]interfaces,InvocationHandler h)

例如:

package com.imooc.jdkproxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class TimeHandler implements InvocationHandler {

    public TimeHandler(Object target) {
        super();
        this.target = target;
    }

    private Object target;
    /**
     * 参数:
     * proxy 被代理的对象
     * method 被代理的方法
     *  args 方法参数
     *  
     *  返回值 Object对象
     */
    @Override
    public Object invoke(Object arg0, Method method, Object[] args)
            throws Throwable {
        long startTime = System.currentTimeMillis();
        System.out.println("汽车开始行使。。。");
        method.invoke(target);
        long endTime = System.currentTimeMillis();
        System.out.println("汽车结束行使。。。 汽车行使时间:" +(endTime - startTime) + "毫秒");
        return null;
    }

}

 

package com.imooc.proxy;

public interface Moveable {
    void move();
}
 

package com.imooc.proxy;

import java.util.Random;

public class Car implements Moveable {

    @Override
    public void move() {
        //实现开车
        try {
            Thread.sleep(new Random().nextInt(1000));
            System.out.println("汽车行使中。。。");
        } catch (InterruptedException e) {
            
            e.printStackTrace();
        }
    }

}
 

import java.lang.reflect.Proxy;

import com.imooc.proxy.Car;
import com.imooc.proxy.Moveable;

public class Test {
    public static void main(String[] args) {
        Car car = new Car();
        InvocationHandler h = new TimeHandler(car);
        Class<?> cls = car.getClass();
        /**
         * arg0 被代理类的类加载器
         * arg1 实现接口
         * arg2 invocationHandler
         */
        Moveable m =  (Moveable)Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(), h);
        m.move();
    }
    
}

2.4 使用cglib动态产生代理

2.4.1 JDK动态代理与CGLIB动态代理区别

JDK动态代理:

(1)只能代理实现了接口的类

(2)没有实现接口的类不能实现JDK的动态代理。

CGLIB动态代理

(1)针对类来实现代理的。

(2)对指定目标类产生一个子类,通过方法拦截技术拦截所有父类方法的调用。

CGLIB动态代理实现原理可参考博客:https://blog.csdn.net/yhl_jxy/article/details/80633194

 

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章