代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式,即通过代理对象访问目标对象。这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。
这里使用到编程中的一个思想:不要随意去修改别人已经写好的代码或者方法,如果需改修改,可以通过代理的方式来扩展该方法。
举个例子来说明代理的作用:假设我们想邀请一位明星,那么并不是直接连接明星,而是联系明星的经纪人,来达到同样的目的。明星就是一个目标对象,他只要负责活动中的节目,而其他琐碎的事情就交给他的代理人(经纪人)来解决,这就是代理思想在现实中的一个例子。
静态代理在使用时,需要定义接口或者父类,被代理对象与代理对象一起实现相同的接口或者是继承相同父类。
public interface IMarry {
void doMarry();
}
public class MarryPerson implements IMarry {
public void doMarry() {
System.out.println("!!!举行婚礼了!!!");
}
}
package com.shenmazong.proxy;
/**
* @program: server-java-demo
* @description: MarryPersonProxy
* @author: 亮子说编程
* @create: 2020-10-15 12:06
**/
public class MarryPersonProxy implements IMarry {
private IMarry target;
public MarryPersonProxy(MarryPerson person) {
this.target = person;
}
public void doMarry() {
doUnknow();
doLove();
target.doMarry();
doLeave();
doUnknow();
}
private void doUnknow() {
System.out.println("我们是陌生人");
}
private void doLove() {
System.out.println("我们相爱了");
}
private void doLeave() {
System.out.println("我们离婚了");
}
}
public class MarryApplication {
public static void main(String[] args) {
MarryPerson marryPerson = new MarryPerson();
MarryPersonProxy marryPersonProxy = new MarryPersonProxy(marryPerson);
marryPersonProxy.doMarry();
}
}
静态代理总结:
优点:
缺点:
动态代理有以下特点:
JDK中生成代理对象的API
代理类所在包:java.lang.reflect.Proxy
JDK实现代理只需要使用newProxyInstance方法,但是该方法需要接收三个参数,完整的写法是:
public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h)
注意该方法是在Proxy类中是静态方法,且接收的三个参数依次为:
ClassLoader loader,
:指定当前目标对象使用类加载器,获取加载器的方法是固定的Class<?>[] interfaces,
:目标对象实现的接口的类型,使用泛型方式确认类型InvocationHandler h
:事件处理,执行目标对象的方法时,会触发事件处理器的方法,会把当前执行目标对象的方法作为参数传入总结:
public interface IMarry {
void doMarry();
}
public class MarryPerson implements IMarry {
public void doMarry() {
System.out.println("!!!举行婚礼了!!!");
}
}
public class MarryInvocationHandler implements InvocationHandler {
Object target;
public MarryInvocationHandler(Object target) {
this.target = target;
}
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("Before invoke " + method.getName());
method.invoke(target, args);
System.out.println("After invoke " + method.getName());
return null;
}
}
public class MarryInvocationApplication {
public static void main(String[] args) {
// 是否保存生成的代理类class文件,默认false不保存
System.getProperties().setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
IMarry marry = new MarryPerson();
InvocationHandler handler = new MarryInvocationHandler(marry);
IMarry proxyHello = (IMarry) Proxy.newProxyInstance(marry.getClass().getClassLoader(), marry.getClass().getInterfaces(), handler);
proxyHello.doMarry();
}
}