代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问。代理类负责为委托类预处理消息,过滤消息并转发消息,以及进行消息被委托类执行后的后续处理。
为了保持行为的一致性,代理类和委托类通常会实现相同的接口,所以在访问者看来两者没有丝毫的区别。通过代理类这中间一层,能有效控制对委托类对象的直接访问,也可以很好地隐藏和保护委托类对象,同时也为实施不同控制策略预留了空间,从而在设计上获得了更大的灵活性。Java 动态代理机制以巧妙的方式近乎完美地实践了代理模式的设计理念。
【JDK方式的动态代理】
package com.xcl.common;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class ProxyFactory implements InvocationHandler {
private Object targetObj;
public Object createProxyInstance(Object targetObj) {
this.targetObj = targetObj;
return Proxy.newProxyInstance(this.targetObj.getClass().getClassLoader(),
this.targetObj.getClass().getInterfaces(),this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
UserServiceBean bean = (UserServiceBean) this.targetObj;
Object result = null;
if (bean.getUser() != null && !"".equals(bean.getUser())) {
//把方法的调用委派给目标对象
result = method.invoke(this.targetObj, args);
}
return result;
}
}
class UserServiceBean implements UserService {
private String user = null;
public String getUser() {
return user;
}
public void setUser(String user) {
this.user = user;
}
public UserServiceBean() {
}
public UserServiceBean(String user) {
this.user = user;
}
@Override
public String getUserName(Integer personId) {
System.out.println("我是getUserName方法");
return "xxx";
}
@Override
public String saveUser(String user) {
System.out.println("我是saveUser方法");
return "yyy";
}
}
interface UserService {
public String getUserName(Integer personId);
public String saveUser(String user);
}
【cglib方式的动态代理】
package com.xcl.common;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class CglibProxyFactory implements MethodInterceptor {
private Object targetObj;//目标类的对象
public Object createProxyInstance(Object targetObj) {
this.targetObj = targetObj;
Enhancer enhancer = new Enhancer();//代理对象
enhancer.setSuperclass(this.targetObj.getClass());//设置代理对象的父类,继承了这个目标类,非final的方法都被覆盖
enhancer.setCallback(this);
return enhancer.create();
}
@Override
public Object intercept(Object proxy, Method method, Object[] args,
MethodProxy methodProxy) throws Throwable {
//环绕通知
InformationServiceBean bean = (InformationServiceBean) this.targetObj;
Object result = null;
if (bean.getUser() != null && !"".equals(bean.getUser())) {
// 把方法的调用委派给目标对象
beforeAdvice();
try {
result = method.invoke(this.targetObj, args);
afterAdvice();
} catch (Exception e) {
exceptionAdvice();
} finally {
finallyAdvice();
}
}
return result;
}
private void finallyAdvice() {
System.err.println("我是finallyAdvice");
}
private void exceptionAdvice() {
System.err.println("我是exceptionAdvice");
}
private void afterAdvice() {
System.err.println("我是afterAdvice");
}
private void beforeAdvice() {
System.err.println("我是beforeAdvice");
}
}
class InformationServiceBean {
private String user = null;
public String getUser() {
return user;
}
public void setUser(String user) {
this.user = user;
}
public InformationServiceBean() {
}
public InformationServiceBean(String user) {
this.user = user;
}
public String getUserName(Integer personId) {
System.out.println("我是getUserName方法");
return "xxx";
}
public String saveUser(String user) {
System.out.println("我是saveUser方法");
return "yyy";
}
}
【测试动态代理】
package com.xcl.common;
import org.junit.After;
import org.junit.AfterClass;
import org.junit.Before;
import org.junit.BeforeClass;
import org.junit.Test;
public class ProxyTest {
@BeforeClass
public static void setUpBeforeClass() throws Exception {
}
@AfterClass
public static void tearDownAfterClass() throws Exception {
}
@Before
public void setUp() throws Exception {
}
@After
public void tearDown() throws Exception {
}
@Test public void proxyTest() {
ProxyFactory factory = new ProxyFactory();
UserService service = (UserService) factory.createProxyInstance(new UserServiceBean("xy"));
service.saveUser("abc");
}
@Test public void cglibProxyTest() {
CglibProxyFactory factory = new CglibProxyFactory();
InformationServiceBean service = (InformationServiceBean) factory.createProxyInstance(new InformationServiceBean("abc"));
service.saveUser("abc");
}
}
动态代理的核心代码
package com.xcl.common;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class DynamicProxyFactory implements InvocationHandler {
private Object targetObj;
public Object createProxyInstance(Object targetObj) {
this.targetObj = targetObj;
return Proxy.newProxyInstance(this.targetObj.getClass()
.getClassLoader(), this.targetObj.getClass().getInterfaces(),
this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
UserServiceBean bean = (UserServiceBean) this.targetObj;
Object result = null;
if (bean.getUser() != null && !"".equals(bean.getUser())) {
// 把方法的调用委派给目标对象
result = method.invoke(this.targetObj, args);
}
return result;
}
}
分享到:
相关推荐
在编程领域,动态代理和工厂方法是两种非常重要的设计模式,它们在软件开发中起着至关重要的作用。这里我们将深入探讨这两种模式,并结合提供的文件名,即“005_Factory_Series_DesignPattern”和“006_Dynamic_...
在结合工厂模式实现动态代理时,我们可以创建一个工厂类,该工厂负责根据需求生成动态代理对象。例如,当需要对数据访问层(DAO)进行日志记录或者事务管理时,可以通过工厂生成带有这些额外功能的代理DAO对象。这样...
这个压缩包中包含的实例源码涵盖了多种设计模式,包括工厂模式、动态代理和责任链模式。让我们深入探讨这些设计模式的核心概念、用途以及它们在实际编程中的应用。 1. 工厂模式:这是最基础的设计模式之一,属于...
- `Proxy`类是JDK动态代理的工厂,它提供静态方法`newProxyInstance()`用于创建代理对象。 - 创建代理对象需要三个参数:类加载器、目标接口数组和InvocationHandler实例。 二、CGLIB代理 CGLIB(Code Generation...
`java.lang.reflect.Proxy` 类是用于创建动态代理对象的工厂类。我们需要使用`Proxy.newProxyInstance()`方法,传入类加载器、接口数组以及自定义的`InvocationHandler`实例,来生成代理对象。 5. **动态代理示例*...
Java动态代理是Java编程中一个重要的特性,它允许我们在运行时创建代理类,这些代理类可以作为原有类的代理,从而在不修改原有类代码的情况下,扩展或增强原有类的功能。这里我们将深入探讨两个主要的Java动态代理...
1. **Java.lang.reflect.Proxy**:这是JDK动态代理的关键类,它提供了创建动态代理对象的工厂方法`newProxyInstance()`。这个方法接收三个参数:`ClassLoader`用于加载代理类,`Interface[]`表示代理对象需要实现的...
JDK动态代理的示例可能包括创建一个代理工厂,生成代理对象并实现InvocationHandler。CGLIB示例可能涉及使用Enhancer类来生成代理对象。对于拦截器,可能会展示如何定义和注册拦截器,以及如何构建拦截器链。此外,...
在动态代理的实现中,Proxy 类是生成代理对象的工厂,它可以生成代理对象。Proxy 类的 newProxyInstance 方法是生成代理对象的入口,该方法需要三个参数:类加载器、被代理对象的接口和 InvocationHandler 接口的...
`Proxy` 类是用于创建动态代理对象的工厂,而`InvocationHandler`接口定义了代理对象调用方法时的行为。当我们创建一个代理对象时,需要提供一个实现了`InvocationHandler`的实例,并指定一组接口,代理对象将实现...
`Proxy`类是Java动态代理的工厂,通过`newProxyInstance()`方法创建代理对象。该方法需要传入三个参数:类加载器、目标对象实现的接口列表和`InvocationHandler`实例。 4. **面向切面编程(AOP)**: AOP是一种...
- `Proxy`类是Java提供的动态代理的工厂,它可以根据一个接口生成实现了该接口的代理类的对象。这个代理类会动态地生成并实现接口的所有方法。 - `InvocationHandler`接口定义了一个处理方法调用的回调方法`invoke...
除了基本的代理实现,JDK动态代理还可以结合其他设计模式,如工厂模式,创建更加复杂的代理对象。此外,Spring框架中的AOP功能也是基于JDK动态代理或CGLIB实现的,它允许开发者定义切面,对满足特定条件的方法进行...
1. **Proxy类**: 是生成动态代理类的工厂类,`Proxy.newProxyInstance()`方法用于创建代理对象,需要传入目标接口、类加载器和InvocationHandler。 2. **InvocationHandler接口**: 定义了一个`invoke()`方法,当通过...
在Java中,我们可以使用JDK的动态代理或者Spring AOP来实现代理模式。 JDK动态代理主要依赖于`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口。Proxy类是生成代理对象的工厂,而...
单例设计模式案例演示 单例模式,是一种常用的软件设计模式。通过单例模式可以保证系统中,**应用该模式的这个类只有一个实例**。即一个类只有一个对象实例。 #### 单例设计模式实现步骤 ...动态代理案例演示
Java中,静态代理和动态代理(Java Proxy API)是常见的实现方式。 4. **工厂模式**:这是一种创建型设计模式,它提供了一种创建对象的最佳方式。工厂模式抽象了实例化过程,用户无需知道具体类的实现细节,只需...
`Proxy`类是动态代理的工厂,它提供了创建动态代理对象的方法。`Proxy.newProxyInstance()`是关键方法,它需要三个参数:代理接口的类加载器、代理接口的Class数组以及一个实现了`InvocationHandler`接口的对象。这...
`Proxy`类是Java动态代理的工厂,它提供了`newProxyInstance()`方法,用于创建代理对象。这个方法需要三个参数:目标接口的类加载器、目标接口数组以及一个`InvocationHandler`实例。`InvocationHandler`接口定义了...