韩顺平设计模式之代理模式

目录

一、代理模式(Proxy)

基本介绍

 二、静态代理

2.1 基本介绍

2.2 应用实例

2.3 静态代理优缺点 

三、动态代理

3.1 基本介绍

3.2 JDK中生成代理对象的 API

3.3 动态代理应用实例

四、Cglib代理

4.1 基本介绍

4.2 Cglib代理模式实现步骤

4.3 Cglib代理模式应用实例

五、几种常见的代理模式介绍—几种变体


一、代理模式(Proxy)

基本介绍

(1) 代理模式:为一个对象提供一个替身,以控制对这个对象的访问。即通过代理对象访问目标对象.这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。

(2)被代理的对象可以是远程对象、创建开销大的对象或需要安全控制的对象。

(3)代理模式有不同的形式, 主要有三种 静态代理动态代理 (JDK代理、接口代理)和 Cglib代理 (可以在内存动态的创建对象,而不需要实现接口,他是属于动态代理的范畴) 。

(4) 代理模式示意图

 二、静态代理

2.1 基本介绍

静态代理在使用时,需要定义接口或者父类,被代理对象(即目标对象)与代理对象一起实现相同的接口或者是继承相同父类。

2.2 应用实例

(1)定义一个接口:ITeacherDao

(2)目标对象 TeacherDAO实现接口 ITeacherDAO

(3)使用静态代理方式,就需要在代理对象 TeacherDAOProxy中也实现 ITeacherDAO

(4)调用的时候通过调用代理对象的方法来调用目标对象.

(5)特别提醒:代理对象与目标对象要实现相同的接口,然后通过调用相同的方法来调用目标对象的方法

思路分析图解(类图)

 

代码实现 

package com.atguigu.proxy.staticproxy;

//接口
public interface ITeacherDao {
    public void teach();//授课的方法
}
package com.atguigu.proxy.staticproxy;

/**
 * @author wjd
 * @date 2021年09月21日 0:10
 */
public class TeacherDao implements ITeacherDao {

    @Override
    public void teach() {
        System.out.println(" 老师授课中... ");
    }
}
package com.atguigu.proxy.staticproxy;

/**
 * @author wjd
 * @date 2021年09月21日 0:11
 */
//代理对象,静态代理
public class TeacherDaoProxy implements ITeacherDao {
    ITeacherDao target;//目标对象,通过接口来聚合

    //构造器
    public TeacherDaoProxy(ITeacherDao target) {
        this.target = target;
    }

    @Override
    public void teach() {
        System.out.println("静态代理开始,完成某些操作...");//方法
        target.teach();
        System.out.println("静态代理提交...");//方法
    }
}
package com.atguigu.proxy.staticproxy;

/**
 * @author wjd
 * @date 2021年09月21日 0:14
 */
public class Client {
    public static void main(String[] args) {
        //创建目标对象(被代理对象)
        TeacherDao teacherDao = new TeacherDao();

        //创建代理对象, 同时将被代理对象传递给代理对象
        TeacherDaoProxy teacherDaoProxy = new TeacherDaoProxy(teacherDao);

        //通过代理对象,调用到被代理对象的方法
        //即:执行的是代理对象的方法,代理对象再去调用目标对象的方法
        teacherDaoProxy.teach();
    }
}

2.3 静态代理优缺点 

(1)优点:在不修改目标对象的功能前提下, 能通过代理对象对目标功能扩展。

(2) 缺点:因为代理对象需要与目标对象实现一样的接口,所以会有很多代理类。

(4)一旦接口增加方法,目标对象与代理对象都要维护。

三、动态代理

3.1 基本介绍

(1)代理对象,不需要实现接口,但是目标对象要实现接口,否则不能用动态代理。

(2) 代理对象的生成,是利用 JDK的 API,动态的在内存中构建代理对象。

(3)动态代理也叫做:JDK代理、接口代理。

3.2 JDK中生成代理对象的 API

(1)代理类所在包:java.lang.reflect.Proxy

(2) JDK实现代理只需要使用 newProxyInstance方法,但是该方法需要接收三个参数,完整的写法是:

static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h )

3.3 动态代理应用实例

应用实例要求

将前面的静态代理改进成动态代理模式(即:JDK代理模式)

思路图解(类图)

代码实现

package com.atguigu.proxy.dynamicproxy;

//接口
public interface ITeacherDao {
    public void teach();//授课的方法
    public void sayHello(String name);
}
package com.atguigu.proxy.dynamicproxy;

/**
 * @author wjd
 * @date 2021年09月21日 0:10
 */
public class TeacherDao implements ITeacherDao {

    @Override
    public void teach() {
        System.out.println(" 老师授课中... ");
    }

    @Override
    public void sayHello(String name) {
        System.out.println("Hello," + name);
    }
}
package com.atguigu.proxy.dynamicproxy;

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

/**
 * @author wjd
 * @date 2021年09月21日 0:28
 */
public class ProxyFactory {
    //维护一个目标对象(被代理对象),Object
    Object target;

    //构造器,对target进行初始化
    public ProxyFactory(Object target) {
        this.target = target;
    }

    //给目标对象(被代理对象)生成一个代理对象
    public Object getProxyInstance() {
        //说明
		/*
		 *  public static Object newProxyInstance(ClassLoader loader,
                                          Class<?>[] interfaces,
                                          InvocationHandler h)

            //1. ClassLoader loader : 指定当前目标对象使用的类加载器, 获取加载器的方法固定
            //2. Class<?>[] interfaces: 目标对象实现的接口类型,使用泛型方法确认类型
            //3. InvocationHandler h : 事情处理,执行目标对象的方法时,会触发事情处理器方法, 会把当前执行的目标对象方法作为参数传入
		 */
        return Proxy.newProxyInstance(target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        System.out.println("JDK代理开始~~");
                        //反射机制调用目标对象的方法
                        Object returnVal = method.invoke(target, args);
                        System.out.println("JDK代理提交");
                        return returnVal;
                }
        });
    }
}
package com.atguigu.proxy.dynamicproxy;

/**
 * @author wjd
 * @date 2021年09月21日 0:37
 */
public class Client {
    public static void main(String[] args) {
        //创建目标对象
        ITeacherDao target = new TeacherDao();

        //给目标对象,创建代理对象, 可以转成 ITeacherDao
        ITeacherDao proxyInstance = (ITeacherDao) new ProxyFactory(target).getProxyInstance();

        // proxyInstance=class com.sun.proxy.$Proxy0 内存中动态生成了代理对象
        System.out.println("proxyInstance=" + proxyInstance.getClass());

        //通过代理对象,调用目标对象的方法
        proxyInstance.teach();
        proxyInstance.sayHello("Jack");
    }
}

四、Cglib代理

4.1 基本介绍

(1) 静态代理和 JDK代理模式都要求目标对象是实现一个接口,但是有时候目标对象只是一个单独的对象,并没有实现任何的接口,这个时候可使用目标对象子类来实现代理-这就是 Cglib代理。

(2) Cglib代理也叫作子类代理,它是在内存中构建一个子类对象从而实现对目标对象功能扩展, 有些书也将Cglib代理归属到动态代理。

(3)Cglib是一个强大的高性能的代码生成包,它可以在运行期扩展 java类与实现 java接口.它广泛的被许多 AOP的框架使用,例如 Spring AOP,实现方法拦截。

(4)在 AOP编程中如何选择代理模式:

1. 目标对象需要实现接口,用 JDK代理

2. 目标对象不需要实现接口,用 Cglib代理

(5)Cglib包的底层是通过使用字节码处理框架 ASM来转换字节码并生成新的类。

4.2 Cglib代理模式实现步骤

(1) 需要引入 cglib的 jar文件

 (2)在内存中动态构建子类,注意代理的类不能为 final,否则报错

java.lang.IllegalArgumentException:

(3)目标对象的方法如果为 final/static,那么就不会被拦截,即不会执行目标对象额外的业务方法

4.3 Cglib代理模式应用实例

应用实例要求

将前面的案例用 Cglib代理模式实现

思路图解(类图)

package com.atguigu.proxy.cglibproxy;

import com.atguigu.proxy.dynamicproxy.ITeacherDao;

/**
 * @author wjd
 * @date 2021年09月21日 0:10
 */
public class TeacherDao implements ITeacherDao {

    @Override
    public void teach() {
        System.out.println(" 老师授课中, 使用cglib代理,不需要实现接口... ");
    }

    @Override
    public void sayHello(String name) {
        System.out.println("Hello," + name);
    }
}
package com.atguigu.proxy.cglibproxy;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

/**
 * @author wjd
 * @date 2021年09月21日 0:28
 */
public class ProxyFactory implements MethodInterceptor {
    //维护一个目标对象(被代理对象),Object
    Object target;

    //构造器,对target进行初始化
    public ProxyFactory(Object target) {
        this.target = target;
    }

    //返回一个代理对象:  是 target 对象的代理对象
    public Object getProxyInstance() {
        //1. 创建一个工具类
        Enhancer enhancer = new Enhancer();
        //2. 设置父类
        enhancer.setSuperclass(target.getClass());
        //3. 设置回调函数
        enhancer.setCallback(this);
        //4. 创建子类对象,即代理对象
        return enhancer.create();
    }

    //重写  intercept 方法,会调用目标对象的方法
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("Cglib代理模式 ~~ 开始");
        Object returnVal = method.invoke(target, objects);
        System.out.println("Cglib代理模式 ~~ 提交");
        return returnVal;
    }
}
package com.atguigu.proxy.cglibproxy;

/**
 * @author wjd
 * @date 2021年09月21日 1:09
 */
public class Client {
    public static void main(String[] args) {
        //创建目标对象
        TeacherDao target = new TeacherDao();

        //获取到代理对象,并且将目标对象传递给代理对象
        TeacherDao proxyInstance = (TeacherDao) new ProxyFactory(target).getProxyInstance();

        //执行代理对象的方法,触发intecept 方法,从而实现 对目标对象的调用
        proxyInstance.teach();
        proxyInstance.sayHello("Lilei");
    }
}

五、几种常见的代理模式介绍—几种变体

(1)防火墙代理

内网通过代理穿透防火墙,实现对公网的访问。

(2)缓存代理

比如:当请求图片文件等资源时,先到缓存代理取,如果取到资源则 ok,如果取不到资源,再到公网或者数据库取,然后缓存。

(3) 远程代理

远程对象的本地代表,通过它可以把远程对象当本地对象来调用。远程代理通过网络和真正的远程对象沟通信息。

(4)同步代理:主要使用在多线程编程中,完成多线程间同步工作

同步代理:主要使用在多线程编程中,完成多线程间同步工作。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值