代理模式 - Java实现
代理模式
代理模式概念
为其他对象提供一个代理以控制对某个对象的访问。代理类主要负责为委托类(真实对象)预处理消息、过滤消息、传递消息给委托类,代理类不现实具体服务,而是利用委托类来完成服务,并将执行结果封装处理。
其实就是代理类为被代理类预处理消息、过滤消息并在此之后将消息转发给被代理类,之后还能进行消息的后置处理。代理类和被代理类通常会存在关联关系(即上面提到的持有的被代理对象的引用),代理类本身不实现服务,而是通过调用被代理类中的方法来提供服务。
静态代理
创建一个接口,然后创建被代理的类实现该接口并且实现该接口中的抽象方法。之后再创建一个代理类,同时使其也实现这个接口。在代理类中持有一个被代理对象的引用,而后在代理类方法中调用该对象的方法。
接口:
public interface HelloInterface {
void sayHello();
}
被代理类:
public class Hello implements HelloInterface{
@Override
public void sayHello() {
System.out.println("Hello zhanghao!");
}
}
代理类:
public class HelloProxy implements HelloInterface{
private HelloInterface helloInterface = new Hello();
@Override
public void sayHello() {
System.out.println("Before invoke sayHello" );
helloInterface.sayHello();
System.out.println("After invoke sayHello");
}
}
代理类调用:被代理类被传递给了代理类HelloProxy,代理类在执行具体方法时通过所持用的被代理类完成调用。
public static void main(String[] args) {
HelloProxy helloProxy = new HelloProxy();
helloProxy.sayHello();
}
输出:
Before invoke sayHello
Hello zhanghao!
After invoke sayHello
使用静态代理很容易就完成了对一个类的代理操作。但是静态代理的缺点也暴露了出来:由于代理只能为一个类服务,如果需要代理的类很多,那么就需要编写大量的代理类,比较繁琐。
动态代理
利用反射机制在运行时创建代理类。
动态代理:
-
特点:字节码随用随创建,随用随加载
-
作用:不修改源码的基础上对方法增强
-
分类:
- 基于接口的动态代理
- 基于子类的动态代理
如何创建代理对象:
前提配置:
接口类:
public interface IProducer {
void saleProduct(float money);
void afterService(float money);
}
实现类:( 被代理类 )
public class Producer implements IProducer {
public void saleProduct(float money){
System.out.println("拿钱销货 " + money);
}
public void afterService(float money){
System.out.println("售后服务 " + money);
}
}
基于接口的动态代理
使用Proxy类中的newProxyInstance方法 | 提供者:JDK
创建代理对象的要求:被代理类最少实现一个接口,如果没有则不能使用
方法的参数:
-
ClassLoader:类加载器。
用于加载代理对象的字节码,和被代理对象使用相同的类加载器。
固定写法:
被代理类.getClass().getClassLoader() -
Class[]:字节码数组
用于让代理对象和被代理对象有相同方法
固定写法:
被代理类.getClass().getInterfaces() -
InvocationHandler:用于提供增强的代码
描述如何代理。一般都是写一个该接口的实现类,通常情况下都是匿名内部类,但不是必须。
此接口的实现类是谁用谁写
代码实现:
public class Client {
public static void main(String[] args) {
final Producer producer = new Producer();
//基于接口的动态代理
IProducer proxyProducer = (IProducer) Proxy.newProxyInstance(
producer.getClass().getClassLoader(),
producer.getClass().getInterfaces(), new InvocationHandler() {
/**
* 作用:执行被代理对象的任何接口方法都会经过该方法
* @param proxy 代理对象的引用
* @param method 当前执行的方法
* @param args 当前执行方法所需的参数
* @return 和被代理对象有相同的返回值
* @throws Throwable
*/
public Object invoke(Object proxy, Method method,
Object[] args) throws Throwable {
// return method.invoke(producer,args);//什么都没干
//提供增强的代码
Object returnValue = null;
//1、获取方法执行的参数
float money = (Float) args[0];
//2、判断当前方法是不是销售
if("saleProduct".equals(method.getName())){
returnValue = method.invoke(producer, money*0.8f);
}
return returnValue;
}
});
proxyProducer.saleProduct(1000f);
}
}
基于子类的动态代理
使用的类:Enhancer | 提供者:第三方cglib库 | 创建代理对象:Enhancer 类中的 create方法
要求:被代理类不能是最终类
方法的参数:
-
Class : 字节码
用于指定被代理对象的字节码
-
Callback:
用于提供增强的代码
一般使用的都是
Callback接口的子接口实现类:MethodInterceptor(翻译:方法拦截)

public class Client {
public static void main(String[] args) {
final Producer producer = new Producer();
//基于子类的动态代理
Producer cglibProducer = (Producer)Enhancer.create(
producer.getClass(), new MethodInterceptor() {
/**
* 执行被代理对象的任何方法都会经过该方法
* @param proxy
* @param method
* @param args
* 以上三个参数和基于接口中的动态代理中 invoke 方法的参数是一样的
* @param methodProxy 当前执行方法的代理对象
* @return
* @throws Throwable
*/
public Object intercept(Object proxy, Method method,
Object[] args, MethodProxy methodProxy)
throws Throwable {
//提供增强的代码
Object returnValue = null;
//1、获取方法执行的参数
float money = (Float) args[0];
//2、判断当前方法是不是销售
if ("saleProduct".equals(method.getName())) {
returnValue = method.invoke(producer, money * 0.8f);
}
return returnValue;
}
});
cglibProducer.saleProduct(10000f);
}
}
基于子类的动态代理不需要被代理类继承一个接口

浙公网安备 33010602011771号