Java代理模式

代理模式

  23中(Gof)中代理模式中的一种,AOP本身就是基于动态代理实现的,所以掌握了代理模式对AOP的学习很有帮助,所以先讲下代理模式

1.静态代理模式

  若代理类在程序运行前就已经存在,那么这种代理方式被成为 静态代理 ,这种情况下的代理类通常都是我们在Java代码中定义的。 通常情况下, 静态代理中的代理类和目标类会实现同一接口或是派生自相同的父类。

1.1 创建接口

/**
* 定义公共接口
* @author dpb[波波烤鸭]
*
*/
public interface SomeService {
   String doSome();
}

1.2 创建接口实现类

/**
* 静态代理模式的实现类
* @author dpb[波波烤鸭]
*
*/
public class SomeServiceImpl implements SomeService{
   @Override
   public String doSome() {
   	// TODO Auto-generated method stub
   	return "hello";
   }
}

1.3 创建静态代理类

/**
* 代理类
*    增强实现类
*    和实现类实现同一个接口
* @author dpb[波波烤鸭]
*
*/
public class SomeProxy implements SomeService{
   private SomeService target;
   public SomeProxy(SomeService target) {
   	super();
   	this.target = target;
   }
   /**
    * 增强方法
    *    将返回结果转换为大写
    */
   @Override
   public String doSome() {
   	return target.doSome().toUpperCase();
   }
}

1.4 测试实现

   public static void main(String[] args) {
   	SomeService some = new SomeServiceImpl();
   	// 获取静态代理对象
   	SomeProxy proxy = new SomeProxy(some);
   	System.out.println(proxy.doSome());
   }

在这里插入图片描述

2.动态代理模式

  代理类在程序运行时创建的代理方式被成为 动态代理。 也就是说,这种情况下,代理类并不是在Java代码中定义的,而是在运行时根据我们在Java代码中的“指示”动态生成的。

代理类型使用场景
JDK动态代理如果目标对象实现了接口,采用JDK的动态代理
cglib动态代理如果目标对象没有实现了接口,必须采用CGLIB动态代理

2.1.JDK动态代理

声明接口
/**
 * 定义公共接口
 * @author dpb[波波烤鸭]
 *
 */
public interface SomeService {
	String doSome();
}
创建实现类
/**
 * 静态代理模式的实现类
 * @author dpb[波波烤鸭]
 *
 */
public class SomeServiceImpl implements SomeService{
	@Override
	public String doSome() {
		// TODO Auto-generated method stub
		return "hello";
	}
}
JDK代理实现
public static void main(String[] args) {
	SomeService some = new SomeServiceImpl();
	// 获取静态代理对象
	SomeService proxy = (SomeService) Proxy.newProxyInstance(
			some.getClass().getClassLoader() // 实现类的类加载器
			, some.getClass().getInterfaces() // 实现类 实现的所有的接口
			, new InvocationHandler()  // 处理器
			{
				/**
				 * proxy 代理对象
				 * method 目标的方法
				 * args 目标方法的参数
				 */
				@Override
				public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
					String msg = (String) method.invoke(some, args);
					return msg.toUpperCase();
				}
			});
	System.out.println(proxy.doSome());

}

在这里插入图片描述

2.2CGLIB动态代理

导入cglib的jar包

在这里插入图片描述

创建目标类
/**
 * 静态代理模式的实现类
 * @author dpb[波波烤鸭]
 *
 */
public class SomeServiceImpl {
	
	public String doSome() {
		// TODO Auto-generated method stub
		return "hello";
	}
}
创建cglib工厂
/**
 * cglib代理类
 * @author dpb[波波烤鸭]
 *
 */
public class CglibProxy implements MethodInterceptor{
	
	private SomeServiceImpl target;
	
	public CglibProxy(SomeServiceImpl target) {
		this.target = target;
	}

	/**
	 * 创建cglib代理对象的方法
	 * @return
	 */
	public SomeServiceImpl createProxy(){
		// 创建增强器
		Enhancer e = new Enhancer();
		// 指定父类
		e.setSuperclass(SomeServiceImpl.class);
		// 指定回调接口对象
		e.setCallback(this);
		// 创建cglib代理对象
		return (SomeServiceImpl) e.create();
	}

	/**
	 * 拦截方法
	 */
	@Override
	public Object intercept(Object obj, Method method, Object[] agrs, MethodProxy proxy) throws Throwable {
		String msg = (String)method.invoke(target, agrs);
		return msg.toUpperCase();
	}

}

测试
public static void main(String[] args) {
	// 定义目标对象
	SomeServiceImpl some = new SomeServiceImpl();
	// 获取代理对象
	SomeServiceImpl proxy = new CglibProxy(some).createProxy();
	System.out.println(proxy.doSome());
}

在这里插入图片描述

Java代理模式是一种结构型设计模式,其目的是为其他对象提供一种代理以控制对这个对象的访问。代理对象可以在客户端和目标对象之间充当中介,以便于客户端访问目标对象时,可以在不改变目标对象的情况下添加一些额外的功能,比如安全性、远程访问、缓存等。 在Java中,代理模式可以通过两种方式实现:静态代理和动态代理。静态代理需要手动编写代理类,而动态代理可以在运行时通过反射机制动态生成代理类,更加灵活。 举个例子,假设我们有一个接口`Subject`,其中定义了一些方法。我们希望在调用这些方法时,增加一些额外的日志记录功能。我们可以编写一个代理类`SubjectProxy`,在代理类中实现接口方法并调用目标对象的方法,同时在方法前后添加日志记录的代码。客户端则通过代理类访问目标对象。 静态代理示例代码如下: ```java public interface Subject { void doSomething(); } public class RealSubject implements Subject { @Override public void doSomething() { System.out.println("RealSubject do something."); } } public class SubjectProxy implements Subject { private Subject realSubject; public SubjectProxy(Subject realSubject) { this.realSubject = realSubject; } @Override public void doSomething() { System.out.println("Before do something."); realSubject.doSomething(); System.out.println("After do something."); } } public class Client { public static void main(String[] args) { Subject realSubject = new RealSubject(); Subject subjectProxy = new SubjectProxy(realSubject); subjectProxy.doSomething(); } } ``` 动态代理示例代码如下: ```java public class SubjectHandler implements InvocationHandler { private Object target; public SubjectHandler(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("Before " + method.getName()); Object result = method.invoke(target, args); System.out.println("After " + method.getName()); return result; } } public class Client { public static void main(String[] args) { Subject realSubject = new RealSubject(); InvocationHandler handler = new SubjectHandler(realSubject); Subject subjectProxy = (Subject) Proxy.newProxyInstance(realSubject.getClass().getClassLoader(), realSubject.getClass().getInterfaces(), handler); subjectProxy.doSomething(); } } ``` 无论是静态代理还是动态代理代理模式都可以在不改变目标对象的情况下,为其添加额外的功能,提高代码的可复用性和灵活性。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

波波烤鸭

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值