【java项目实战】代理模式(Proxy Pattern),静态代理 VS 动态代理

摘要:
定义代理模式是对象的结构型模式,代理模式给某一个对象提供了一个代理对象,并由代理对象控制对原对象的引用。分类静态代理和动态代理静态代理静态代理类图代码演示样例接口packagecom.liang.pattern;publicinterfaceUserManager{publicvoidaddUser;publicvoiddelUser;publicvoidmodifyUser;publicStringfindUser;}目标对象packagecom.liang.pattern;publicclassUserManagerImplimplementsUserManager{publicvoidaddUser{try{System.out.println;}catch{e.printStackTrace();thrownewRuntimeException();}}publicvoiddelUser{System.out.println;}publicStringfindUser{System.out.println;return"于亮";}publicvoidmodifyUser{System.out.println;}}代理类,我们使用代理对象做一些日志记录,我们将简略的打印信息到控制台。packagecom.liang.pattern;publicclassUserManagerImplProxyimplementsUserManager{privateUserManageruserManager;publicUserManagerImplProxy{this.userManager=userManager;}publicvoidaddUser{//记录日志等操作或打印输入參数System.out.println;try{userManager.addUser;//运行成功。

这篇博文,我们主要以类图和代码的形式来对照学习一下静态代理和动态代理。重点解析各自的优缺点。

定义

代理模式(Proxy Pattern)是对象的结构型模式,代理模式给某一个对象提供了一个代理对象,并由代理对象控制对原对象的引用。

代理模式不会改变原来的接口和行为,仅仅是转由代理干某件事,代理能够控制原来的目标,比如:代理商,代理商仅仅会买东西,但并不会改变行为。不会制造东西。

让我们通过以下的代码好好理解一下这句话。

分类

静态代理和动态代理

静态代理

静态代理类图

【java项目实战】代理模式(Proxy Pattern),静态代理 VS 动态代理第1张

代码演示样例

【java项目实战】代理模式(Proxy Pattern),静态代理 VS 动态代理第2张

接口

package com.liang.pattern;

public interface UserManager {

	public void addUser(String userId,String userName);
	
	public void delUser(String userId);
	
	public void modifyUser(String userId,String userName);
	
	public String findUser(String userId);
	
}

目标对象
package com.liang.pattern;

public class UserManagerImpl implements UserManager {

	public void addUser(String userId, String userName) {

		try{
			System.out.println("UserManagerImpl.addUser() userId-->>" + userId);
		}catch(Exception e){
			e.printStackTrace();
			
			throw new RuntimeException();
		}
		
		
	}

	public void delUser(String userId) {

		System.out.println("UserManagerImpl.delUser() userId-->>" + userId);
	}

	public String findUser(String userId) {

		System.out.println("UserManagerImpl.findUser() userId-->>" + userId);
		return "于亮";
	}

	public void modifyUser(String userId, String userName) {

		System.out.println("UserManagerImpl.modifyUser() userId-->>" + userId);
	}

}

代理类,我们使用代理对象做一些日志记录,我们将简略的打印信息到控制台。
package com.liang.pattern;

public class UserManagerImplProxy implements UserManager {

	private UserManager userManager;
	public UserManagerImplProxy(UserManager userManager){
		this.userManager = userManager;
	}
	public void addUser(String userId, String userName) {
                //记录日志等操作或打印输入參数
		System.out.println("start-->>addUser() userId-->>" + userId);
		try{

			userManager.addUser(userId, userName);
                //运行成功。打印成功信息
			System.out.println("success-->>addUser()");
		}catch(Exception e){
			e.printStackTrace();
                //失败时。打印失败信息
			System.out.println("error-->>addUser()");
			//throw new RuntimeException();
		}
	}

	public void delUser(String userId) {
		//同上,略
		userManager.delUser(userId);
	}

	public String findUser(String userId) {
		//同上。略
		userManager.findUser(userId);
		return null;
	}

	public void modifyUser(String userId, String userName) {
		//同上。略
		userManager.modifyUser(userId, userName);
		
	}

}

client调用
package com.liang.pattern;

public class Client {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		UserManager userManager = new UserManagerImplProxy(new UserManagerImpl());
		userManager.addUser("001","于亮");
	}
} 

输出结果,此方法运行成功
start-->>addUser() userId-->>001
UserManagerImpl.addUser() userId-->>001
success-->>addUser()

从类图我们能够看出。client本来能够直接和目标对象打交道,代理中间加了一个间接层。他们实现的功能是一样的,也没有改变參数。相信大家对上面的类图和代码非常熟悉。跟我们平时看别人的博文一样。没有不论什么差别。以下我们看一下静态代理的优缺点。

优缺点

长处

1、直观感受,静态代理是实实在在的存在的,我们自己写的。

2、在编译期增加,提前就指定好了谁调用谁,效率高。

缺点

相同,它的长处也成了它致命的缺点。

1、静态代理非常麻烦。须要大量的代理类

当我们有多个目标对象须要代理时,我就须要建立多个代理类。改变原有的代码,改的多了就非常有可能出问题,必须要又一次測试。

2、反复的代码会出如今各个角落里,违背了一个原则:反复不是好味道

我们应该杜绝一次次的反复。

3、在编译期增加,系统的灵活性差

我们能够看到代理类的每一个方法中,都有记录日志,运行成功或失败的代码,每一个方法都反复了一遍,假设我们须要改动的话,并没有比不用静态代理时降低改动的地方。仅仅是不用改动目标类。动态代理非常好的为我们攻克了这个问题。以下我们看一下动态代理。

动态代理

动态代理类图

【java项目实战】代理模式(Proxy Pattern),静态代理 VS 动态代理第3张

代码演示样例

【java项目实战】代理模式(Proxy Pattern),静态代理 VS 动态代理第4张

代理类(不明确,就看看凝视吧)

package com.liang.pattern;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
 * 採用JDK动态代理必须实现InvocationHandler接口。採用Proxy类创建对应的代理类
 * @author liang
 *
 */
public class ProxyHandler implements InvocationHandler {

	private Object targetObject;
	/**
	 * 目标的初始化方法。依据目标生成代理类
	 * @param targetObject
	 * @return
	 */
	public Object newProxyInstance(Object targetObject){
		this.targetObject = targetObject;
		//第一个參数,目标的装载器
		//第二个參数,目标接口,为每一个接口生成代理
		//第三个參数,调用实现了InvocationHandler的对象。当你一调用代理,代理就会调用InvocationHandler的invoke方法
		return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this);
	}
	
	/**
	 * 反射,这样你能够在不知道详细的类的情况下,依据配置的參数去调用一个类的方法。

在灵活编程的时候很实用。 */ public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //记录日志等操作或打印输入參数 System.out.println("start-->>" + method.getName()); for(int i=0;i<args.length;i++){ //打印调用目标方法的參数 System.out.println(args[i]); } Object ret = null; try{ //调用目标方法 ret = method.invoke(targetObject, args); //运行成功。打印成功信息 System.out.println("success-->>" + method.getName()); }catch(Exception e){ e.printStackTrace(); //失败时,打印失败信息 System.out.println("error-->>" + method.getName()); throw e; } return ret; } }


client调用
package com.liang.pattern;

public class Client {

	/**
	 * @param args
	 */
	public static void main(String[] args) {

		ProxyHandler proxyHandler = new ProxyHandler();
		UserManager userManager = (UserManager)proxyHandler.newProxyInstance(new UserManagerImpl());
		
		String name = userManager.findUser("0001");
		System.out.println("client.main-->>" + name);
	}

} 

输出结果,执行成功

start-->>findUser
0001
UserManagerImpl.findUser() userId-->>0001
success-->>findUser
client.main-->>于亮

接口和目标类,同上,我就不再浪费大家的带宽了。

优缺点


长处

1、一个动态代理类更加简单了,能够解决创建多个静态代理的麻烦,避免不断的反复多余的代码

2、调用目标代码时,会在方法“执行时”动态的增加,决定你是什么类型,才调谁,灵活

缺点

1、系统灵活了。可是相比而言,效率减少了,比静态代理慢一点

2、动态代理比静态代理在代码的可读性上差了一点,不太easy理解

3、JDK动态代理仅仅能对实现了接口的类进行代理

总结

静态代理VS动态代理,打成了平手,各自有各的独特之处,均不可取代,在项目中究竟使用哪种代理,没有最好。仅仅有更合适。

免责声明:文章转载自《【java项目实战】代理模式(Proxy Pattern),静态代理 VS 动态代理》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇终于干掉了默认的输入法, 关于ctfmon.exe文件浅谈 linux 多线程编程和 windows 多线程编程的异同下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Java 面向切面编程(Aspect Oriented Programming,AOP)

本文内容 实例 引入 原始方法 装饰者模式 JDK 动态代理和 cglib 代理 直接使用 AOP 框架——AspectWerkz 最近跳槽了,新公司使用了 AOP 相关的技术,于是查点资料,复习一下。之前,多少知道点,但没怎么在实际项目中使用过~ 下载 demo 实例 引入 package com.cap.aop;   publi...

domino控制台命令

Broadcast message 向该服务器的用户广播消息 Broadcast message usernames 向该服务器的某一用户广播消息 Dbcache Flush 关闭当前在数据库高速缓存中打开的所有数据库。使用此命令可以在高速缓存中维护数据库,例如:备份或恢复数据库。 Drop username 关闭一个或多个服务器会话。如果...

如何调试 Android 上 HTTP(S) 流量

转自: http://greenrobot.me/devpost/how-to-debug-http-and-https-traffic-on-android/ 如何调试 Android 上 HTTP(S) 流量 前面的话 在Android开发中我们常常会和API 打交道,可能你不想,但是这是避不开的。大部分情况下,调试发送网络请求和接收响应的过程都是...

maven全局配置文件settings.xml详解

概要 settings.xml有什么用? 如果在Eclipse中使用过Maven插件,想必会有这个经验:配置settings.xml文件的路径。settings.xml文件是干什么的,为什么要配置它呢?从settings.xml的文件名就可以看出,它是用来设置maven参数的配置文件。并且,settings.xml是maven的全局配置文件。而pom.xm...

理解 Spring(二):AOP 的概念与实现原理

什么是 AOP AOP 的基本术语 Spring AOP 的简单应用 Spring AOP 与动态代理 Spring AOP 的实现原理(源码分析) 扩展:为什么 JDK 动态代理要求目标类必须实现接口 什么是 AOP AOP(Aspect Oriented Programming,面向切面编程)是一种编程范式,它是对 OOP(Object O...

Java学习之二-Java反射机制

问题: 在运行时,对一个JAVA类,能否知道属性和方法;能否调用它的任意方法? 答案是可以的,JAVA提供一种反射机制可以实现。 目录 什么是JAVA的反射机制 JDK中提供的Reflection API JAVA反射机制提供了什么功能 获取类的Class对象 获取类的Fields 获取类的Method 获取类的Constructor 新建类的实例 ...