深入理解JAVA中的代理模式

时间:2022-11-18 23:03:23

前言

代理是什么

事故现场:我家的宠物今天生病了,而我又没有相关的医学知识,所以我只让我的宠物多喝热水吗?

结果显然是不行的,所以我需要去找宠物医生这些更专业的人来帮我的宠物治病。

这个时候,代理就出现了,而宠物医生就是代理,而我就是目标对象。

总结起来就是代理代替目标对象执行相关操作,即是对目标对象的一种功能扩展。

使用代理模式的条件

1、两个角色:执行者,被代理对象

2、注重过程,必须要做,被代理的对象没时间做或者不想做,不专业

3、执行者必须拿到被代理对象的个人资料

1.静态代理

代码实现:

/**
* 我和宠物医生都是人,都有治疗技能,但是宠物医生比我更专业
*/
interface IPerson{
void treat(Pet pet); //治疗技能
}
/**
* 宠物类
*/
class Pet{
private String name;
public Pet(String name){
this.name = name;
}
public String getName() {
return name;
}
} /**
* 目标对象实现”IPerson“接口
*/
class Self implements IPerson{
private Pet pet;
public Self(Pet pet){
this.pet = pet;
}
public void treat(Pet pet){
System.out.println(pet.getName() + ",你要多喝点水");
}
} /**
* 代理对象与目标对象实现同一接口
*/
class PetDoctor implements IPerson{
//接收目标对象
private IPerson targetObj;
public PetDoctor(IPerson targetObj){
this.targetObj = targetObj;
}
@Override
public void treat(Pet pet) {
System.out.println("对" + pet.getName() + "进行检查");
targetObj.treat(pet);
System.out.println("对" + pet.getName() + "进行治疗");
}
}

代码测试:

public static void main(String[] args){
//我的宠物
Pet pet = new Pet("多多");
//目标对象
IPerson target = new Self(pet);
//代理对象
IPerson proxy = new PetDoctor(target);
proxy.treat(pet);
}

运行结果:

宠物医生对多多进行检查
我对多多说,你要多喝点水
宠物医生对多多进行治疗

结果很明显,医生比我更专业,我只会让我的宠物喝水,但医生会先检查再进行专业的治疗,所以说代理是让更专业的对象帮你做事。

2.动态代理

动态代理又分为jdk动态代理和cglib动态代理,两者的区别是jdk动态代理的实现是基于接口,而cglib动态代理是基于继承,但两者做的是同一件事,那就是字节码重组

基本流程都是根据目标对象的资料,通过反射获取该对象的信息,然后根据信息按照特定的写法重写一个java类,再进行编译并动态加载到JVM中运行,所以说动态代理在底层其实就是实现了字节码重组。

jdk动态代理实例演示

Person接口

//定义Person接口,技能是煮饭
public interface Person {
void cook();
}

我自己,也就是被代理的对象,但我只会做可乐鸡翅

public class Oneself implements Person {
@Override
public void cook() {
System.out.println("我会做可乐鸡翅");
}
}

动态代理类,也是一个厨师,因为初始对于做菜比我更专业

public class Kitchener implements InvocationHandler{
//需要代理的目标对象
private Object object; public Kitchener(Object object){
this.object = object;
} @Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("我会做糖醋排骨");
method.invoke(object,args); //这是我会做的,其余两样是代理对象初始会做的
System.out.println("我会做九转大肠");
return null;
}
}

测试代码

public class TestJdk {
public static void main(String[] args){
//创建目标代理对象
Oneself oneself = new Oneself(); InvocationHandler kitchener = new Kitchener(oneself);
/*
* 通过Proxy的newProxyInstance方法来创建我们的代理对象,做的就是字节码重组的工作,新生成一个java类在编译再加载到JVM运行
* 第一个参数是类加载器
* 第二个参数是我们这里为代理对象提供的接口,也就是代理对象所实现的接口,所以说在jdk动态代理中被代理对象需要实现一个接口
* 第三个参数handler, 我们这里将这个代理对象关联到了上方的 InvocationHandler 这个对象上
*/
Person proxy = (Person) Proxy.newProxyInstance(kitchener.getClass().getClassLoader(),
oneself.getClass().getInterfaces(),
kitchener);
System.out.println(proxy.getClass()); // (1)
proxy.cook(); }
}

测试结果

class com.sun.proxy.$Proxy0    //(2)
我会做糖醋排骨
我会做可乐鸡翅
我会做九转大肠

可以看到(1)行代码打印出来的是一个代理类,而代理对象通过生成java类再编译加载运行对用户来说是无感知的,我们只知道返回回来的是一个代理对象,然后由代理对象去帮我们做事。

而cglib代理的实现原理也是一样的,只不过一个是基于接口,一个是基于继承,原理都是通过反射获取对象信息再根据对象信息创建java类编译加载运行,所以cglib暂时就不展开了,后期可以自己手写一个动态加深理解。

学习了动态代理后,在本人的工作中是没使用过的,但却是了解spring的AOP实现的必要基础,因为spring的AOP实现就是基于动态代理实现的。