在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。
在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。 代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式;即通过代理对象访问目标对象。这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。
这里使用到编程中的一个思想:不要随意去修改别人已经写好的代码或者方法,如果需改修改,可以通过代理的方式来扩展该方法。
通常我们在实例研发工作中,遇到的场景是:要访问的对象在远程的机器上。在面向对象系统中,有些对象由于某些原因(比如某些操作需要安全控制),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层。
- 何时使用:想在访问一个类时做一些控制。
- 如何解决:增加中间层。
-
静态代理
静态代理在使用时,需要定义接口被代理对象与代理对象一起实现相同的接口。
1.接口
/**
* 接口
*/
public interface IGoodsDao extends BaseMapper<Goods>{
int insert(Goods goods);
}
2.实际使用的GoodsDao
/**
* dao的实现代码
*/
public class GoodsDaoImp implements IGoodsDao {
public int insert(Goods goods) {
dbMapper.insert(goods);
}
}
3.代理对象:DaoProxy
public class DaoProxy implements IGoodsDao {
//接收保存目标对象
private IGoodsDao goodsDao ;
public DaoProxy(IGoodsDaogoodsDao goodsDao){
this.goodsDao =goodsDao ;
}
/**
* 商品的保存
*/
public void save() {
goodsDao.save();//此处执行目标的方法
}
}
综上所述:可以做到在不修改目标对象的功能前提下,对目标功能扩展. 面向修改关闭,面向扩展开放的原则。
动态代理:最常用的 Java 反射机制
JDK 代理
代理类是实现了一个 InvocationHandler 的接口,我们通过 reflect.Proxy 的类的 newProxyInstance 方法就可以得到这个接口的实例,然后再来作为参数传递进去,这里每一个在代理类上处理的东西也会被重定向到调用处理器上。
Cglib 代理
应用场景:JDK 中的动态代理通过反射类 Proxy 和 InvocationHandler 回调接口实现,要求委托类必须实现一个接口,只能对该类接口中定义的方法实现代理,这在实际编程中有一定的局限性。
使用 CGLib 实现动态代理,并不要求委托类必须实现接口,底层采用 asm 字节码生成框架生成代理类的字节码