深入浅出java静态代理和动态代理

首先介绍一下,什么是代理:

代理模式,是常用的设计模式。特征是,代理类与委托类有相同的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类。以及事后处理消息。

代理类和委托类,存在着关联关系。代理类的对象本身并不真正实现服务,知识通过调用委托类的对象的相关方法。

代理类可以分为两种:静态代理和动态代理。

静态代理:

代理类是由程序员创建,或由工具生成的代码 编译成的。在程序运行前,代理类的 *.class文件已经存在了。直接就可以运行 。

动态代理:

动态代理的代理类。没有直接由源代码生成。动态代理类的对象是在程序运行时由JAVA反射机制动态生成,不需要手工编写源代码。从而提高了软件的可扩展性。JAVA反射机制可以生成任意类型的动态代理类。

静态代理的实现:

接口:

package test.static.pattern;

public interface UserManager {

	public void addUser(String userId,String userName);

	public void delUser(String userId);

	public String findUser(String userId);

	public void modifyUser(String userId,String userNameString);
}

实现类:

package test.static.pattern;

public class UserManagerImpl implements UserManager {

	@Override
	public void addUser(String userId, String userName) {
		System.out.println("UserManagerImpl.addUser()  userId-->>" + userId);
	}

	@Override
	public void delUser(String userId) {
		System.out.println("UserManagerImpl.delUser() userId-->>" + userId);
	}

	@Override
	public String findUser(String userId) {
		System.out.println("UserManagerImpl.findUser() userId-->>" + userId);
		return null;
	}

	@Override
	public void modifyUser(String userId, String userNameString) {
		System.out.println("UserManagerImpl.modifyUser() userId-->>" + userId);
	}

}

静态代理类:(只持有对象的引用)

package test.static.pattern;

public class UserManagerImplProxy implements UserManager {

	private UserManager userManager;

	public UserManagerImplProxy	(UserManager userManager){
		this.userManager=userManager;
	}

	@Override
	public void addUser(String userId, String userName) {
		userManager.addUser(userId, userName);
	}

	@Override
	public void delUser(String userId) {
		userManager.delUser(userId);
	}

	@Override
	public String findUser(String userId) {

		return userManager.findUser(userId);
	}

	@Override
	public void modifyUser(String userId, String userNameString) {
		userManager.modifyUser(userId, userNameString);
	}

}

客户端:

package test.static.pattern;

public class Client {
	/**
	 * @param args
	 */
	public static void main(String[] args) {
		//正常思路:客户端直接实例化出 子类的对象
		UserManager userManager=new UserManagerImpl();
		//静态代理:客户端实例化代理,通过代理取 子类的引用
		UserManager userManager=new UserManagerImplProxy(new UserManagerImpl());
		userManager.addUser("0001", "张三");
	}
}

执行结果:

静态代理,因为代理类持有对象的引用,所以可以对其进行控制。

现象1:对于各个功能模块来说,都要建立对应的代理类。造成大量的代理类

现象2:但是如果对各个实现相同的控制,则需要重复写大量的代码。

为了避免重复代码出现多次,我们接着看什么是动态代理。

动态代理的实现:

接口:

package test.dynamic.pattern;

public interface UserManager {

	public void addUser(String userId,String userName);

	public void delUser(String userId);

	public String findUser(String userId);

	public void modifyUser(String userId,String userNameString);
}

实现类:

package test.dynamic.pattern;

public class UserManagerImpl implements UserManager {

	@Override
	public void addUser(String userId, String userName) {
		System.out.println("UserManagerImpl.addUser()  userId-->>" + userId);
	}

	@Override
	public void delUser(String userId) {
		System.out.println("UserManagerImpl.delUser() userId-->>" + userId);
	}

	@Override
	public String findUser(String userId) {
		System.out.println("UserManagerImpl.findUser() userId-->>" + userId);
		return null;
	}

	@Override
	public void modifyUser(String userId, String userNameString) {
		System.out.println("UserManagerImpl.modifyUser() userId-->>" + userId);
	}

}

动态代理生成类:

package test.dynamic.pattern;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

import org.omg.CORBA.SystemException;
import org.omg.CORBA.portable.InputStream;
import org.omg.CORBA.portable.InvokeHandler;
import org.omg.CORBA.portable.OutputStream;
import org.omg.CORBA.portable.ResponseHandler;

public class LogHandler implements InvocationHandler {

	/**
	 * 调用引用对象类的方法抽象
	 */
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {

		Object ret=null;
		try {
			//调用目标方法
			ret=method.invoke(targetObject, args);
		} catch (Exception e) {
			throw e;
		}
		return null;
	}

	//目标实现的引用
	private Object targetObject;

	/**
	 * 代理类生成方法
	 * @param targetObject
	 * @return
	 */
	public Object newProxyInstance(Object targetObject){
		this.targetObject=targetObject;
		return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this);
	}
}

客户端:

package test.dynamic.pattern;

public class Client {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		//动态代理
		LogHandler logHandler=new LogHandler();
		UserManager userManager=(UserManager)logHandler.newProxyInstance(new UserManagerImpl());
		//userManager.addUser("0001", "张三");
		userManager.delUser("111");
	}
}

执行结果:

动态代理类,在程序中没有体现。只有在程序运行的时候采用创建相应的代理类。这样就可以少些大量的代理类。对于现象2,相同控制代码是如何减少的。请看下一篇博客。

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-11-03 20:58:40

深入浅出java静态代理和动态代理的相关文章

Java深入浅出系列(四)——深入剖析动态代理--从静态代理到动态代理的演化

静态代理 如上图,在程序执行之前.程序猿就要编写Proxy.然后进行编译,即在程序执行之前,代理类的字节码文件就已经生成了(Proxy类的class文件已经存在了). 静态代理尽管在增强现有的接口业务功能方面有非常大长处,可是大量使用这样的静态代理,会使系统内的类的规模大量增大,不易维护.而且Proxy类和RealSubject类功能本质上是一样的.仅仅只是Proxy起到了一个中介的作用,这样的代理在系统中的存在导致了系统结构的臃肿和松散. 为了解决问题.产生了动态代理.动态代理是在系统执行中,

【Java】代处理?代理模式 - 静态代理,动态代理

>不用代理 有时候,我希望在一些方法前后都打印一些日志,于是有了如下代码. 这是一个处理float类型加法的方法,我想在调用它前打印一下参数,调用后打印下计算结果.(至于为什么不直接用+号运算,见[Java]Float计算不准确) package com.nicchagil.study.java.demo.No09代理.No01不用代理; import java.math.BigDecimal; public class FloatCalculator { public float add(fl

java静态代理与动态代理简单分析

原创作品,可以转载,但是请标注出处地址http://www.cnblogs.com/V1haoge/p/5860749.html 1.动态代理(Dynamic Proxy) 代理分为静态代理和动态代理,静态代理是在编译时就将接口.实现类.代理类一股脑儿全部手动完成,但如果我们需要很多的代理,每一个都这么手动的去创建实属浪费时间,而且会有大量的重复代码,此时我们就可以采用动态代理,动态代理可以在程序运行期间根据需要动态的创建代理类及其实例,来完成具体的功能. 其实方法直接调用就可以完成功能,为什么

Java基础:静态代理和动态代理

转载请注明出处:jiq?钦's technical Blog 一.静态代理: 假设原来有一个实现了指定接口/抽象类的子类: class RealSubject implements Subject{ public void request(){ System.out.print("real request handling\n"); } } 现在有两种情况会发生: 1新的代码需要调用Subject接口,但是需要给每个接口加入新代码(比如日志记录.权限控制等): 2旧的代码已经使用了Su

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

这篇博文,我们主要以类图和代码的形式来对照学习一下静态代理和动态代理.重点解析各自的优缺点. 定义 代理模式(Proxy Pattern)是对象的结构型模式,代理模式给某一个对象提供了一个代理对象,并由代理对象控制对原对象的引用. 代理模式不会改变原来的接口和行为,仅仅是转由代理干某件事,代理能够控制原来的目标,比如:代理商,代理商仅仅会买东西,但并不会改变行为.不会制造东西. 让我们通过以下的代码好好理解一下这句话. 分类 静态代理和动态代理 静态代理 静态代理类图 代码演示样例 接口 pac

Java设计模式学习06——静态代理与动态代理(转)

原地址:http://blog.csdn.net/xu__cg/article/details/52970885 一.代理模式 为某个对象提供一个代理,从而控制这个代理的访问.代理类和委托类具有共同的父类或父接口,这样在任何使用委托类对象的地方都可以使用代理类对象替代.代理类负责请求的预处理.过滤.将请求分配给委托类处理.以及委托类处理完请求的后续处理. 二.代理模式结构 UML类图: 由上图代理模式的结构为: 抽象角色: 真实对象和代理对象的共同接口. 代理角色: 代理对象角色内部含有对真实对

Java:静态代理and动态代理

代理模式是常用的设计模式,其特征是代理类与委托类具有相同的接口,在具体实现上,有静态代理和动态代理之分.代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务,也就是说代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等. 静态代理和动态代理的一个显著区别: 静态代理:由程序员创建或特定工具自动生成源代码,再对其编译.在程序运行前,代理类的.class文件就

JAVA学习篇--静态代理VS动态代理

本篇博客的由来,之前我们学习大话设计,就了解了代理模式,但为什么还要说呢? 原因: 1,通过DRP这个项目,了解到了动态代理,认识到我们之前一直使用的都是静态代理,那么动态代理又有什么好处呢?它们二者的区别是什么呢? 2,通过学习动态代理了解到动态代理是一种符合AOP设计思想的技术,那么什么又是AOP? 下面是我对它们的理解! 代理Proxy: Proxy代理模式是一种结构型设计模式,主要解决的问题是:在直接访问对象时带来的问题 代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对

java 代理模式(静态代理、动态代理、Cglib代理) 转载

Java的三种代理模式 1.代理模式 代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式;即通过代理对象访问目标对象.这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能.这里使用到编程中的一个思想:不要随意去修改别人已经写好的代码或者方法,如果需改修改,可以通过代理的方式来扩展该方法 代理模式最大的特点就是代理类和实际业务类实现同一个接口(或继承同一父类),代理对象持有一个实际对象的引用,外部调用时操作的是代理对象,而在代理对象的内部实现中又会去调