设计模式之(二)代理模式:模拟java范型中的代理模式
引言
因为要使用java中的范型,所以需要学会代理模式打基础;就比如要学习Retrofit网络框架,首先要学会建造者模式打基础。换句话说,Retrofit是设计模式之一:建造者模式的最佳试炼场;而同样java中的范型是代理模式的最佳试炼场。话不多说,静态、动态代理模式都要学习起来!
简介
在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。
在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。
详细介绍
意图:为其他对象提供一种代理以控制对这个对象的访问。
主要解决:
在直接访问对象时带来的问题,比如说:要访问的对象在远程的机器上。在面向对象系统中,有些对象由于某些原因(比如对象创建开销很大,或者某些操作需要安全控制,或者需要进程外的访问),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层。
何时使用:想在访问一个类时做一些控制。
如何解决:增加中间层。
关键代码:实现与被代理类组合。
应用实例:
1、Windows 里面的快捷方式。
2、买房子或者租房子,这里的中介就是代理。
3、买火车票不一定在火车站买,也可以去代售点。
4、一张支票或银行存单是账户中资金的代理。支票在市场交易中用来代替现金,并提供对签发人账号上资金的控制。
5、spring aop。
优点: 1、职责清晰。 2、高扩展性。 3、智能化。
缺点:
1、由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。
2、实现代理模式需要额外的工作,有些代理模式的实现非常复杂。
使用场景:按职责来划分,通常有以下使用场景:
1、远程代理。
2、虚拟代理。
3、Copy-on-Write 代理。
4、保护(Protect or Access)代理。
5、Cache代理。
6、防火墙(Firewall)代理。
7、同步化(Synchronization)代理。
8、智能引用(Smart Reference)代理。
注意事项:
1、和适配器模式的区别:适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口。
2、和装饰器模式的区别:装饰器模式为了增强功能,而代理模式是为了加以控制。
实践一:静态代理
步骤1:创建服务类接口
/**
* @data on 2020/10/30 4:28 PM
* @auther armStrong
* @describe 静态代理-创建服务类接口
*/
public interface BuyHouse {
void buyHosue();
}
步骤2:创建接口实现类
/**
* @data on 2020/10/30 4:29 PM
* @auther 接口实现类
* @describe 干买房这件事儿
*/
public class BuyHouseImpl implements BuyHouse {
@Override
public void buyHosue() {
System.out.println("我要买房");
}
步骤3:创建实现接口的静态代理类
/**
* @data on 2020/10/30 4:31 PM
* @auther armStrong
* @describe 中介,做代理帮客户买房子(一条龙服务)
*/
public class BuyHouseProxy implements BuyHouse {
public BuyHouse bHosue;
public BuyHouseProxy(final BuyHouse buyhosue){
this.bHosue = buyhosue;
}
@Override
public void buyHosue() {
System.out.println("买房前准备");
bHosue.buyHosue();
System.out.println("买房后装修");
}
}
步骤4:模拟实现静态代理
public class Case60 extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_case60);
//调用静态代理方法
staticProxy();
}
//静态代理
private void staticProxy() {
//自己买房
BuyHouse buyHouse = new BuyHouseImpl();
buyHouse.buyHosue();
//代理帮助买房
BuyHouseProxy buyHouseProxy = new BuyHouseProxy(buyHouse);
buyHouseProxy.buyHosue();
}
}
实践二:动态代理
代替实践一中的步骤3,不用创建静态代理类,创建处理器类交给JDK在运行时帮我们进行动态创建代理对象。
步骤3:创建动态代理处理器类
/**
* @data on 2020/10/30 4:55 PM
* @auther armStrong
* @describe 动态代理处理器类
*/
public class DynamicProxyHandler implements InvocationHandler {
private Object object;
public DynamicProxyHandler(final Object object) {
this.object = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("买房前准备");
Object result = method.invoke(object, args);
System.out.println("买房后装修");
return result;
}
}
步骤4:模拟实现动态代理
public class Case60 extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_case60);
//调用实现静态代理
// staticProxy();
//调用实现动态代理jdk
dynamicProxy();
//CGLIB代理 //不写了
}
//动态代理(JDK方式)
//无需手动创建代理类对象,真正的代理对象由JDK再运行时为我们动态的来创建。
private void dynamicProxy() {
BuyHouse buyHouse = new BuyHouseImpl();
BuyHouse proxyBuyHouse = (BuyHouse) Proxy.newProxyInstance(BuyHouse.class.getClassLoader(), new Class[]{BuyHouse.class}, new DynamicProxyHandler(buyHouse));
proxyBuyHouse.buyHosue();
}
}