Spring中的代理模式
什么是代理模式
在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。
在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。
静态代理
角色分析:
- 抽象角色 :一般会使用接口或者抽象类来解决
- 真实角色 :被代理的角色
- 代理角色 :代理真实角色 , 代理真实角色后,我们一般会做一些附属操作
- 客户 : 访问代理对象的人!
代码分析(以租房为例)
(1)接口
//租房
public interface Rent {
public void rent();
}
(2)真实角色
//房东
public class Host implements Rent {
public void rent() {
System.out.println("房东要出租房子!");
}
}
(3)代理角色
public class Proxy implements Rent {
private Host host;
public Proxy() {
}
public Proxy(Host host) {
this.host = host;
}
public void rent() {
seeHouse();
host.rent();
}
//看房
public void seeHouse(){
System.out.println("中介带你看房");
}
(4)客户端访问代理角色
public class Client {
public static void main(String[] args) {
//房东要租房子
Host host = new Host();
//代理,中介帮房东租房子,但是呢?代理角一般会有一些附属操作!
Proxy proxy = new Proxy(host);
//你不用面对房东,直接找中介租房即可!
proxy.rent();
}
}
(5)分析
在以上程序中,出租房对应的是Rent接口,房东是真实角色,他需要出租房子,也就是实现Rent接口,并实现rent方法。
中介属于代理角色,需要实现Rent接口并实现rent方法,但是他可以实现一些额外的操作比如seehouse()
客户就是租客
(6)优缺点
优点: 1、职责清晰。 2、高扩展性。 3、智能化。
缺点:
1、由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。
2、实现代理模式需要额外的工作,有些代理模式的实现非常复杂。一个真实角色就需要一个代理角色
3、必须提前知道你要代理的是什么
动态代理
- 动态代理和静态代理角色一样
- 动态代理的代理类是动态生成的,不是我们直接写好的!
- 动态代理分为两大类 : 基于接口的动态代理 , 基于类的动态代理
- 基于接口 — JDK 动态代理 【我们在这里使用】
- 基于类 : cglib
- java字节码实现 : javasist
代码实现
(1)接口
public interface Rent {
public void rent();
}
(2)真实角色
public class Host implements Rent {
public void rent() {
System.out.println("出租房");
}
}
(3)动态生成代理类
public class ProxyInvocationHandler implements InvocationHandler {
//被代理的接口
private Rent rent;
public void setRent(Rent rent) {
this.rent = rent;
}
//生成得到代理类
public Object getProxy(){
return Proxy.newProxyInstance(this.getClass().getClassLoader(),
rent.getClass().getInterfaces(),this);
}
//处理代理实例并返回结果
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//动态代理的本质就是反射机制的实现
seehouse();
Object result = method.invoke(rent, args);
return result;
}
public void seehouse(){
System.out.println("中介看房子");
}
}
(4)客户访问代理角色
public class Client {
public static void main(String[] args) {
//真实角色
Host host = new Host();
ProxyInvocationHandler pih = new ProxyInvocationHandler();
pih.setRent(host);
Rent proxy = (Rent) pih.getProxy();
proxy.rent();
}
}