1.静态代理
public class StaticProxy {
public static void main(String[] args) {
//我们无法直接使用被代理对象,只能使用代理对象
//使用代理对象一样可以实现 被代理对象的功能 不过还会额外的增加代理对象的功能
YellowCattle yellowCattle = new YellowCattle();
yellowCattle.salePhone();
}
}
//目标对象(被代理对象)
class PhoneFactory implements Phone {
public int salePhone() {
int price = 200;
System.out.println("手机价格是" + price);
return price;
}
}
//代理对象
class YellowCattle extends PhoneFactory implements Phone {
//这里想调用 被代理对象 有许多方法,不仅仅可以通过继承,也可以把 被代理对象 当成成员属性 关联进来
@Override
public int salePhone() {
//调用目标对象 实现基本功能
int newPrice = super.salePhone();
//自己增强的功能
newPrice += 500;
System.out.println("黄牛哥哥加价之后" + newPrice);
return newPrice;
}
}
//只是编写规范,没有特殊意义
interface Phone {
int salePhone();
}
#######
控制台打印
手机价格是200
黄牛哥哥加价之后700
#######
从上面简单的demo可以感受到,当我们想为一个功能增强,但是又不可以直接在上面修改代码时,就可以使用代理模式去实现。做到代码的相对解耦,既不会影响被代理对象的逻辑,还可以 增加自己额外的功能。但是弊端也非常的明显。如果我现在要对一个新的对象进行代理,那么代理对象也要创建一个全新的进行匹配。比如我有一个卖手机的目标对象,就需要一个卖手机的代理对象,有一个卖橘子的就需要在来一个卖橘子的代理对象。当需要代理的对象很庞时,就会出现代码冗余难以维护,于是动态代理就诞生了。
2.动态代理
动态代理有两种实现方式,第一个是基于java的动态代理,他要求必须有接口才可以实现。第二个是基于cglib的动态代理,他是基于继承实现的,这里先聊聊java的动态代理。
前面我们已经了解到了,静态代理会出现一个问题,就是被代理对象需要一对一的绑定一个代理对象。并且当需要代理的功能发生变化,被代理对象以及代理对象都要修改相关代码,十分难以维护。那么我们的想法就是,可不可以动态的告诉我你需要代理的是那个对象,以及需要代理的是那个方法,我们通过反射来调用呢?于是基于前面的静态代理我们做出调整。
public class DynamicProxy {
public static void main(String[] args) {
//调用者核心代码
//1.首先要有被代理对象
IPhoneSale target = new Target();
//2.其次需要代理对象 并将被代理对象传入代理对象
MyProxy myProxy = new MyProxy(target);
//3.核心代码 必须这样写
// 参数1是 被代理对象的类加载器 参数二是 被代理对象实现的接口信息 参数三是 代理对象
IPhoneSale proxyTarget = (IPhoneSale)Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
myProxy);
//这里调用salePhone反法时,调用的是 代理对象中的invoke方法 从而实现功能增强
int sale = proxyTarget.salePhone();
System.out.println(sale);
}
}
//jdk的动态必须有接口
interface IPhoneSale{
int salePhone();
}
//被代理对象(目标对象)
class Target implements IPhoneSale{
@Override
public int salePhone() {
System.out.println("我是目标方法");
return 100;
}
}
//代理对象必须实现这个接口
class MyProxy implements InvocationHandler{
//被代理对象声明为成员属性 依靠调用者动态传入
Object target = null;
public MyProxy(Object target){
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//核心代码 利用反射会调用目标对象的方法 具体调用那个 是由调用者发起的
Object invoke = method.invoke(target, args);
//增强代码
System.out.println("我是代理对象,我要涨价100!");
if (invoke != null) {
Integer sale = (Integer) invoke + 100;
return sale;
}
return null;
}
}
########
控制台打印:
我是目标方法
我是代理对象,我要涨价100!
200
########
动态代理可以实现代码的解耦,增强灵活度。比如我现在接口中需要代理一个新的方法,那么只需要修改被代理对象的实现以及调用者调用的方法即可。