JAVA--动态代理(基于jdk)

代理模式的理解

  • 官方概念:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用-----《百度百科》
  • 我的理解:代理就是帮别人做事情,就比如生活中的留学中介、厂家代理等等
  • 留学中介(代理):帮助学校去招生,这就是这个学校的代理,功能就是帮助这个学校完成招生工作
  • 代理的特点:
    1. 中介和代理做的东西(功能)的一样的,就是招生
    2. 中介是学校的代理,而学校则是目标
    3. 学生/家长----------中介(学校的介绍,帮助办理入学手续)----------学校
    4. 中介是代理,是要收取费用的
    5. 代理不让你访问到目标
  • 为什么要找代理:
    1. 方便
    2. 学生/家长没有能力去访问学校,或者学校不接受以个人形式的来访
  • 在开发过程中的例子:
    1. 有A类C类A类要调用C类的方法完成某个功能,但C类不允许A类的调用,那么就在A类C类之间创建一个B代理,让A类访问B代理B代理访问C类
    2. 注册账号功能需要手机短信验证码,项目本身的没有这个功能的,只能找代理来完成这个工作(比如说阿里云的信息服务)
禁止访问
A类
C类
B代理
没有此功能
注册账号
发送短信
阿里云信息服务

代理模式的作用

  1. 功能增强:在原有的基础上,增加了某些功能,增加的功能就叫功能增强
  2. 控制访问:代理不允许你访问到目标,例如工厂只能有代理商拿货,而个人不被允许
  3. 调用目标类的方法,并执行

静态代理

  • 理解:代理是增加手动实现的,自己创建java类,表示代理类,同时代理目标也是自己确定的
  • 优点:实现简单、理解容易
  • 缺点:
    1. 当目标类增加,代理类可能需要成倍的增加,代理数量过多
    2. 当接口中的功能增加或者修改,或影响众多的实现类,厂家类和代理类都需要全部修改
  • 例子:模拟一个用户淘宝购买u盘的行为
    • 用户是客户端类
    • 淘宝商家是代理某个U盘的代理类
    • 制造U盘的厂家是目标类
    • 三者的关系:用户---------->淘宝代理商---------->生产商
    • 生产商和代理商完成的功能是一致的,都是买U盘
    • 实现步骤:
    1. 创建一个接口,定义卖U盘的方法,表示生产商和代理商的功能
    2. 创建代理商类实现步骤1的接口
    3. 创建生产商类实现步骤1的接口
    4. 创建用户类,调用代理商购买U盘
    • 代理类完成的功能:
    1. 目标类(生成商类)的方法调用
    2. 功能增强(送满减红包等等)

java反射机制(动态代理用到)

  • Method类,表示方法类中的方法,通过Method可以执行某个类中的某个方法
  • 创建HelloService接口
package com.zsc.test;

public interface HelloService {
    // 根据name打招呼
    public void sayHello(String name);
}
  • 创建HelloServiceImpl接口实现类
package com.zsc.test;
// HelloService接口的实现类
public class HelloServiceImpl implements HelloService {
    @Override
    public void sayHello(String name) {
        System.out.println("您好"+name);
    }
}
  • 创建测试类TestHelloService
package com.zsc.test;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

// 测试
public class TestHelloService {
    public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        // 正常的通过对象调用sayHello方法
        HelloService helloService = new HelloServiceImpl();
        System.out.println("通过对象调用sayHello方法");
        helloService.sayHello("张三");
        // 通过反射机制执行sayHello方法,核心Method(类中的方法)
        // 创建需要调用sayHello的对象
        HelloService Service = new HelloServiceImpl();
        // 获取sayHello名称对应的Method类对象
        // method代表的是sayHello这个方法
        /*
            参数:
            1.需要执行的方法名
            2.参数的类型
         */
        Method method = HelloService.class.getMethod("sayHello", String.class);
        // 通过Method可以执行sayHello方法的调用
        /*
            invoke是Method类中的一个方法,表示执行方法的调用
            参数:
            1.Object:表示对象,就是要执行这个对象的sayHello方法
            2.Object...args:方法执行时的参数值
            返回值:Object,表示返回执行后的返回值,sayHello返回是void
         */
        // 表示执行Service对象的sayHello返回参数是李四
        System.out.println("通过反射调用sayHello方法");
        Object object = method.invoke(Service,"李四");
    }
}
  • 结果在这里插入图片描述

动态代理

  • 理解:使用java的反射机制、创建对象的能力来创建代理对象,而不用自己手动的去创建代理类(java文件);在程序执行时,调用jdk提供的方法才能创建代理类对象
  • 在目标类很多的时候,可以使用动态代理
  • 实现:
    1. jdk的动态代理(理解):使用java反射包中的类和接口实现动态代理的功能;反射包java.lang.reflect,里面包含3个类InvocationHandler、Method、Proxy,一定要有目标类功能的接口
    2. cglib动态代理(了解):cglib是第三方的工具库,cglib的原理是继承,通过继承目标类,创建它的子类,在子类中重新父类中同名的方法,实现功能的修改,从而实现动态代理;因为cglib的原理是继承,所以目标类不能是final,方法也不能是final,cglib对目标类的要求比较宽松,只需要目标类能继承就行了,在很多框架都有所用到,比如说mybatis和spring
  • jdk的动态代理:
    • 反射包java.lang.reflect,里面包含3个类InvocationHandler、Method、Proxy
    • InvocationHandler接口(调用处理器):Invoke()方法,(需要执行的功能,我的理解:生产商或代理商的卖货的功能,及代理商发放的优惠卷的代理类功能增强都属于要执行的功能)
package java.lang.reflect;

public interface InvocationHandler {
	// 表示代理对象执行的功能代码,你的代理类需完成的功能就写在Invoke()中
	/*
		参数:
		Object proxy:jdk创建的代理对象,无需复值
		Method method:目标类中的方法,jdk提供method的对象
		Object args:目标类中方法的参数,jdk提供
	*/
    Object invoke(Object proxy, Method method, Object[] args) throws Throwable;
}
  • Method类:表示目标类的方法,和上面反射的Method类的用法一样 (执行目标类的功能,我的理解:从生产商拿货)
      通过Method可以执行某个目标类的方法,method.invoke(目标对象,方法参数),和InvocationHandler接口的invoke()方法没有任何关系,只是方法名相同而已等同于静态代理中代理执行目标类的方法
public final class Method extends Executable {
	/*
        invoke是Method类中的一个方法,表示执行方法的调用
        参数:
        1.Object obj:表示对象,就是要执行这个对象的sayHello方法
        2.Object...args:方法执行时的参数值
        返回值:Object,表示返回执行后的返回值
    */
	public Object invoke(Object obj, Object... args) throws IllegalAccessException, IllegalArgumentException, InvocationTargetException
}
  • Proxy类:核心对象,创建代理对象,用Proxy类的方法代替new关键字的使用 (创建需要执行代理类方法的对象,我的理解:选择在哪个淘宝代理商中购买U盘
public class Proxy implements Serializable {
	// Proxy类中的静态newProxyInstance(),创建代理对象,等同于 代理类 a = new 代理类()
	/*
		参数:
		ClassLoader loader:类加载器,负责向内存中加载对象,使用反射来获取ClassLoader,类A,A.getClass().getClassLoad()
		Class<?>[] interfaces:接口,目标对象实现的接口,也是通过反射来实现
		InvocationHandler h:上面的InvocationHandle接口,代理类要完成的功能
		返回值:代理类的对象
	*/
	public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
}
  • 实现步骤:
    1. 创建接口,定义目标类需完成的方法(功能)
    2. 创建目标类实现步骤1的接口
    3. 创建InvocationHandler接口的实现类,在invoke方法中实现代理类的方法(功能),就是调用目标类的方法和代理类的功能增强
    4. 使用Proxy类的静态方法newProxyInstance,创建代理对象,并把返回值转为接口类型
  • 具体的代码实现(已在淘宝上买U盘为例)
  1. 创建UsbSell接口
package com.zsc.test.service;
// 目标类实现的方法
public interface UsbSell {
    public float sell(int amount);
}
  1. 创建实现UsbSell接口的目标类UsbKingFactory
package com.zsc.test.factory;

import com.zsc.test.service.UsbSell;
// 目标类
public class UsbKingFactory implements UsbSell {
    // 目标方法
    @Override
    public float sell(int amount) {
        System.out.println("执行目标类方法sell");
        return 85.0f;
    }
}
  1. 创建InvocationHandler接口的实现类
package com.zsc.test.handler;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
// 必须实现InvocationHandler接口,完成代理功能(1.调用目标方法,2.功能增强)
// 负责执行目标类的方法(执行生产商卖U盘sell的这个功能)
// 我的理解:只是执行了目标类的方法和代理类的功能增强,返回目标类方法的返回值给InvocationHandler对象
public class MySellHandle implements InvocationHandler {
    private Object target = null;
    // 动态代理:目标对象是活动的,不是固定的,需要传进来
    // 传入是哪个对象,就给哪个对象创建代理
    public MySellHandle(Object target){
        // 给目标对象赋值(生产商)
        this.target = target;
    }
    @Override
    public Object invoke(Object o, Method method, Object[] objects) throws Throwable {
        Object res = null;
        // 代理商向生产商发送订单,购买生产商的U盘
        // 执行目标方法,返回值是目标类方法的值(及U盘的出厂价)
        res = method.invoke(target,objects);
        // 代理商加价(代理商的利润)
        if (res != null){
            Float price = (Float) res;
            price = price + 25;
            res = price;
        }
        // 代理类实现的功能增强
        System.out.println("代理商返回红包");
        // 返回代理商要销售的价格
        return res;
    }
}
  1. 创建测试类,使用Proxy类的静态方法newProxyInstance,创建代理对象
package com.zsc.test;

import com.zsc.test.factory.UsbKingFactory;
import com.zsc.test.handler.MySellHandle;

import com.zsc.test.service.UsbSell;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

public class MainShop {
    public static void main(String[] args) {
        // 创建代理对象使用Proxy
        // 创建目标(生产商)对象
        UsbSell factory = new UsbKingFactory();
        // 创建InvocationHandler对象
        InvocationHandler handler = new MySellHandle(factory);
        //创建代理对象
        UsbSell proxy =(UsbSell) Proxy.newProxyInstance(factory.getClass().getClassLoader(),
                factory.getClass().getInterfaces(),handler);
        // 通过代理执行方法
        // 下面执行的是proxy在创建时指定的InvocationHandler接口(这里是handler对象)的invoke方法
        float price = proxy.sell(1);
        System.out.println("通过动态代理,代理商的价格为"+price);
    }
}

动态代理的作用(用处)

  • 可以在不改变目标方法的前提下,可以在代理中增强自己的功能代理
  • 比如:在开发过程中,有一个功能是其他人写的(自己并不知道原码),只提供了一个类,需要自己创建对象去调用其中的方法,那么在自己调用的过程中发现有一个方法的功能不能满足项目的需求,这时候就通过代理的方式增强该方法的功能

动态代理的总结

  • 我的理解:动态代理就是在不能修改目标类代理的时候,通过代理的方式增强目标类的方法,但要注意的是在使用jdk动态代理时目标类一个要实现实现功能接口
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值