动态署理在Java中有着遍及的应用,好比Spring AOP,Hibernate数据查询、测试框架的后端mock、RPC,Java注解工具获取等。静态署理的署理干系在编译时就确定了,而动态署理的署理干系是在编译期确定的。静态署理实现简朴,适合于署理类较少且确定的环境,而动态署理则给我们提供了更大的机动性。本日我们来探讨Java中两种常见的动态署理方法:JDK原活跃态署理和CGLIB动态署理。
JDK原活跃态署理
先从直观的示例说起,假设我们有一个接口Hello
和一个简朴实现HelloImp
:
// 接口 interface Hello{ String sayHello(String str); } // 实现 class HelloImp implements Hello{ @Override public String sayHello(String str) { return "HelloImp: " + str; } }
这是Java种再常见不外的场景,利用接口拟定协议,然后用差异的实现来实现详细行为。假设你已经拿到上述类库,假如我们想通过日志记录对sayHello()
的挪用,利用静态署理可以这样做:
// 静态署理方法 class StaticProxiedHello implements Hello{ ... private Hello hello = new HelloImp(); @Override public String sayHello(String str) { logger.info("You said: " + str); return hello.sayHello(str); } }
上例中静态署理类StaticProxiedHello
作为HelloImp
的署理,实现了沟通的Hello
接口。用Java动态署理可以这样做:
// Java Proxy // 1. 首先实现一个InvocationHandler,要领挪用会被转发到该类的invoke()要领。 class LogInvocationHandler implements InvocationHandler{ ... private Hello hello; public LogInvocationHandler(Hello hello) { this.hello = hello; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { if("sayHello".equals(method.getName())) { logger.info("You said: " + Arrays.toString(args)); } return method.invoke(hello, args); } } // 2. 然后在需要利用Hello的时候,通过JDK动态署理获取Hello的署理工具。 Hello hello = (Hello)Proxy.newProxyInstance( getClass().getClassLoader(), // 1. 类加载器 new Class<?>[] {Hello.class}, // 2. 署理需要实现的接口,可以有多个 new LogInvocationHandler(new HelloImp()));// 3. 要领挪用的实际处理惩罚者 System.out.println(hello.sayHello("I love you!"));
运行上述代码输出功效:
日志信息: You said: [I love you!] HelloImp: I love you!
上述代码的要害是Proxy.newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler handler)
要领,该要了解按照指定的参数动态建设署理工具。三个参数的意义如下:
loader
,指定署理工具的类加载器;interfaces
,署理工具需要实现的接口,可以同时指定多个接口;handler
,要领挪用的实际处理惩罚者,署理工具的要领挪用城市转发到这里(*留意1)。newProxyInstance()
会返回一个实现了指定接口的署理工具,对该工具的所有要领挪用城市转发给InvocationHandler.invoke()
要领。领略上述代码需要对Java反射机制有必然相识。动态署理神奇的处所就是:
InvocationHandler.invoke()
要领,在invoke()
要领里我们可以插手任何逻辑,好比修改要领参数,插手日志成果、安详查抄成果等;之后我们通过某种方法执行真正的要领体,示例中通过反射挪用了Hello工具的相应要领,还可以通过RPC挪用长途要领。留意1:对付从Object中担任的要领,JDK Proxy会把
hashCode()
、equals()
、toString()
这三个非接口要领转发给InvocationHandler
,昆山软件开发,其余的Object要领例不会转发。详见JDK Proxy官方文档。
假如对JDK署理后的工具范例举办深挖,可以看到如下信息:
# Hello署理工具的范例信息 class=class jdkproxy.$Proxy0 superClass=class java.lang.reflect.Proxy interfaces: interface jdkproxy.Hello invocationHandler=jdkproxy.LogInvocationHandler@a09ee92