`

代理模式Proxy

    博客分类:
  • Java
阅读更多

一、 代理模式

  代理模式是常用的Java 设计模式,它的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。

按照代理类的创建时期,代理类可分为两种。

 

静态代理类:由程序员创建或由特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。

动态代理类:在程序运行时,运用反射机制动态创建而成。

 

静态代理的小例子:

 

     HelloServiceProxy 类是代理类,HelloServiceImpl类是委托类,这两个类都实现了HelloService接口。其中HelloServiceImpl类是 HelloService接口的真正实现者,而HelloServiceProxy类是通过调用HelloServiceImpl 类的相关方法来提供特定服务的。HelloServiceProxy类的echo()方法和getTime()方法会分别调用被代理的 HelloServiceImpl 对象的echo()方法和getTime()方法,并且在方法调用前后都会执行一些简单的打印操作。

 

       由此可见,代理类可以为委托类预处理消息、把消息转发给委托类和事后处理消息等。

 

 

package proxy;

import java.util.Date;

/**
 *************************************
  * @Title   HelloService.java
  * @Author  张作强
  * @Date    2010-8-12
  * @Comment 代理类和委托类的公共接口
 *************************************
 */
public interface HelloService {
	public String echo(String msg);
	public Date getTime(); 
}

 

 

 

package proxy;
/**
 * 委托类
 */
import java.util.Date;

public class HelloServiceImpl implements HelloService {

	public String echo(String msg) {
		return "echo" + msg;
	}

	public Date getTime() {
		return new Date();
	}
}

 

 

package proxy;
/**
 * 代理类
 */
import java.util.Date;

public class HelloServiceProxy implements HelloService {
	//表示被代理的HelloService 实例
	private HelloService helloService;
	public HelloServiceProxy(HelloService helloService) {
		this.helloService = helloService;
	}
	
	public void setHelloServiceProxy(HelloService helloService){
		this.helloService = helloService;
	}

	public String echo(String msg) {
		//预处理 
		System.out.println("before calling echo()"); 
		//调用被代理的HelloService 实例的echo()方法
		String result = helloService.echo(msg);
		//事后处理
		System.out.println("after calling echo()");
		return result;
	}

	public Date getTime() {
		//预处理
		System.out.println("before calling getTime()"); 
		//调用被代理的HelloService 实例的getTime()方法
		Date date=helloService.getTime(); 
		//事后处理
		System.out.println("after calling getTime()"); 
		return date;
	}

}

 

package proxy;

public class ClientTest {
	public static void main(String[] args) {
		HelloService helloService = new HelloServiceImpl();
		// 注入helloService
		HelloService helloServiceProxy = new HelloServiceProxy(helloService);
		
		System.out.println(helloServiceProxy.echo("Hello World~!"));
		System.out.println("--------- 分割线 ---------");
		System.out.println(helloServiceProxy.getTime());
	}
}

  

 

分享到:
评论
5 楼 csdn_zuoqiang 2010-08-13  
4 楼 csdn_zuoqiang 2010-08-13  
首先我们来明确一下动态代理的定义:一个动态代理类在运行期implements一组interface,使得interface实现类的方法调用被分派至其他的类(另外的interface实现类或者任意的类)的方法。讲得更通俗一些,要了解动态代理,我们就要知道什么东西动态了,代理了什么?首先,一个Proxy代理了一组interface的方法。注意,代理的是interface,而不是Class,也不是abstract Class;其次,Proxy具有的型别由绑定的interface所决定的,动态就体现在此。
3 楼 csdn_zuoqiang 2010-08-12  

dynamic proxy的实战步骤

实际上dynamic proxy只有关键以下几个东西

一、 业务接口:一个Interface
           如:HelloService.java

二、 实现业务接口的类:一个继承Interface的Class
           如:HelloServiceImpl.java

三、 自己写一个继承了java.lang.reflect.InvocationHandler的Handler类(匿名类)
            如:InvocationHandler handler = new InvocationHandler(){}

四、 在这个Handler类中实现invoke()方法
             如:public Object invoke(Object proxy,Method method,Object args[]) throws Throwable {}

五、 在invoke()方法中一定要记得写return method.invoke(wrapped, args)
           如:Object result = method.invoke(helloService,args);
               ......
                 return result;

六、 要使Handler和自己的业务接口关联还的写下面的代码

             如:
Class classType = HelloService.class; 
			return (HelloService)Proxy.newProxyInstance(classType.getClassLoader(),new Class[]{classType},handler); 


2 楼 csdn_zuoqiang 2010-08-12  
三、动态代理类
  与静态代理类对照的是动态代理类,动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。动态代理类不仅简化了编程工作,而且提高了软件系统的可扩展性,因为Java 反射机制可以生成任意类型的动态代理类。java.lang.reflect 包中的Proxy类和InvocationHandler 接口提供了生成动态代理类的能力。
  Proxy类提供了创建动态代理类及其实例的静态方法。
  (1)getProxyClass()静态方法负责创建动态代理类,它的完整定义如下:
  public static Class getProxyClass(ClassLoader loader, Class[] interfaces) throws IllegalArgumentException
  参数loader 指定动态代理类的类加载器,参数interfaces 指定动态代理类需要实现的所有接口。
  (2)newProxyInstance()静态方法负责创建动态代理类的实例,它的完整定义如下:
  public static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler handler) throws
  IllegalArgumentException
参数loader 指定动态代理类的类加载器,参数interfaces 指定动态代理类需要实现的所有接口,参数handler 指定与动态代理类关联的 InvocationHandler 对象。

  以下两种方式都创建了实现Foo接口的动态代理类的实例:
  
/**** 方式一 ****/
  //创建InvocationHandler对象
  InvocationHandler handler = new MyInvocationHandler(...);
  //创建动态代理类
  Class proxyClass = Proxy.getProxyClass(Foo.class.getClassLoader(), new Class[] { Foo.class });
  //创建动态代理类的实例
  Foo foo = (Foo) proxyClass.getConstructor(new Class[] { InvocationHandler.class }).
  newInstance(new Object[] { handler });
  
/**** 方式二 ****/
  //创建InvocationHandler对象
  InvocationHandler handler = new MyInvocationHandler(...);
  //直接创建动态代理类的实例
  Foo foo = (Foo) Proxy.newProxyInstance(Foo.class.getClassLoader(),new Class[] { Foo.class }, handler);

由Proxy类的静态方法创建的动态代理类具有以下特点:
  动态代理类是public、final和非抽象类型的;
  动态代理类继承了java.lang.reflect.Proxy类;
  动态代理类的名字以“$Proxy”开头;
  动态代理类实现getProxyClass()和newProxyInstance()方法中参数interfaces指定的所有接口;
  Proxy 类的isProxyClass(Class cl)静态方法可用来判断参数指定的类是否为动态代理类。只有通过Proxy类创建的类才是动态代理类;
  动态代理类都具有一个public 类型的构造方法,该构造方法有一个InvocationHandler 类型的参数。

由Proxy类的静态方法创建的动态代理类的实例具有以下特点:
  1. 假定变量foo 是一个动态代理类的实例,并且这个动态代理类实现了Foo 接口,那么“foo instanceof Foo”的值为true。把变量foo强制转换为Foo类型是合法的:
  (Foo) foo //合法
  2.每个动态代理类实例都和一个InvocationHandler 实例关联。Proxy 类的getInvocationHandler(Object proxy)静态方法返回与参数proxy指定的代理类实例所关联的InvocationHandler 对象。
  3.假定Foo接口有一个amethod()方法,那么当程序调用动态代理类实例foo的amethod()方法时,该方法会调用与它关联的InvocationHandler 对象的invoke()方法。
  InvocationHandler 接口为方法调用接口,它声明了负责调用任意一个方法的invoke()方法:
  Object invoke(Object proxy,Method method,Object[] args) throws Throwable
  参数proxy指定动态代理类实例,参数method指定被调用的方法,参数args 指定向被调用方法传递的参数,invoke()方法的返回值表示被调用方法的返回值。
1 楼 csdn_zuoqiang 2010-08-12  
动态代理类

package proxy;

/**
 *************************************
  * @Title   HelloServiceProxyFactory.java
  * @Author  张作强
  * @Date    2010-8-12
  * @Comment 创建一个实现了 HelloService 接口的动态代理类的实例
  *          参数 helloService 引用被代理的 HelloService 实例
 *************************************
 */
import java.lang.reflect.*;
public class HelloServiceProxyFactory {
	public static HelloService getHelloServiceProxy(final HelloService helloService){
		//创建一个实现了InvocationHandler接口的匿名类的实例 
		InvocationHandler handler=new InvocationHandler(){ 
			public Object invoke(Object proxy,Method method,Object args[])throws Exception{ 
				System.out.println("before calling "+method); //预处理 
				Object result=method.invoke(helloService,args);
				//调用被代理的HelloService 实例的方法 
				System.out.println("after calling "+method); //事后处理 
				return result; 
				} 
			}; 
			
			@SuppressWarnings("rawtypes")
			Class classType=HelloService.class; 
			return (HelloService)Proxy.newProxyInstance(classType.getClassLoader(), 
					new Class[]{classType}, handler); 
			} 
} 



package proxy;
/**
 *************************************
  * @Title   ClientTest2.java
  * @Author  张作强
  * @Date    2010-8-12
  * @Comment 先创建了一个HelloServiceImpl 实例
  *          然后创建了一个动态代理类实例helloServiceProxy
  *          最后调用动态代理类实例的echo()方法
 *************************************
 */
public class ClientTest2{
	public static void main(String args[]){
		HelloService helloService=new HelloServiceImpl();
		HelloService helloServiceProxy=HelloServiceProxyFactory.getHelloServiceProxy(helloService);
		System.out.println("动态代理类的名称为"+helloServiceProxy.getClass().getName());
		System.out.println(helloServiceProxy.echo("Hello World~!"));
		}
} 



结果:
动态代理类的名称为$Proxy0
before calling public abstract java.lang.String proxy.HelloService.echo(java.lang.String)
after calling public abstract java.lang.String proxy.HelloService.echo(java.lang.String)
echo: Hello World~!

相关推荐

    代理模式 Proxy Pattern

    ### 代理模式 Proxy Pattern #### 概念定义 代理模式是一种结构型设计模式,它允许程序员为某对象创建一个代理对象来控制对该对象的访问。简单来说,就是在不修改原始类的基础上,通过引入代理对象来扩展类的行为...

    设计模式之代理模式proxy

    **设计模式之代理模式(Proxy Pattern)** 设计模式是软件工程中的一种最佳实践,它是在特定情境下解决常见问题的模板。代理模式是其中一种行为设计模式,它的核心思想是为一个对象提供一个替身或者代理,以控制对...

    apache开源项目源码commons-proxy-1.0-src(全部高质量代理模式proxy的java源程序)

    全部高质量代理模式proxy的java源程序 java.proxy,代理模式源码,设计模式,apache开源项目源码commons-proxy-1.0-src 各种代理模式操作的工具类源码以及代理模式案例源码,你会从中得到意想不到的效果! apache...

    设计模式之代理模式proxy.zip

    代理模式是一种常用的设计模式,它在软件开发中扮演着重要的角色。代理模式允许我们为一个对象创建一个代理,这个代理对象在客户端和目标对象之间起到中介的作用,可以增强或控制对目标对象的访问。代理模式的主要...

    设计模式之代理模式Proxy

    代理模式是设计模式中的一种结构型模式,它在对象交互中起到了中介的作用,允许通过代理对象来控制对原对象的访问。代理模式的核心思想是为一个对象提供一个替身,以便增加新的功能或者控制对原对象的访问。这种模式...

    代理模式 proxy-learn.rar

    在`proxy-learn`这个压缩包中,应该包含了这两种代理模式的详细示例代码,供初学者参考学习。对于初学者来说,理解并掌握代理模式是提升编程能力的重要步骤,它能帮助我们更好地理解和应用面向对象设计原则。

    设计模式C++学习之代理模式(Proxy)

    代理模式是一种设计模式,它是结构型模式之一,主要用于在客户端和目标对象之间建立一个代理对象,以便控制对目标对象的访问。在C++中,代理模式可以用来为其他对象提供一种代理以控制对这个对象的访问,或者增加...

    代理模式(Proxy Pattern) 1. 概述 1.1 基本概念 1.2 为什么需要代理模式 1.3 代理模式的四个角色 2. 代理模式的类型 2.1 静态代理 2.2 JDK动态代理

    代理模式(Proxy Pattern) 1. 概述 1.1 基本概念 1.2 为什么需要代理模式 1.3 代理模式的四个角色 2. 代理模式的类型 2.1 静态代理 2.2 JDK动态代理 2.3 CGLIB动态代理 3. 代理模式的UML类图和基本实现 3.1 UML类图...

    c++-设计模式之代理模式(Proxy)

    代理模式(Proxy Pattern)是一种结构型设计模式,用于为其他对象提供一种代理以控制对这个对象的访问。代理模式通常用于保护、延迟加载、记录请求等场景,可以在不改变原有对象的情况下为其增加新的功能。 代理...

    结构型模式之代理模式(Proxy)

    代理模式是一种设计模式,属于结构型模式之一,其主要目的是为其他对象提供一个代理,以控制对该对象的访问。在实际应用中,代理模式能够帮助我们实现如下的功能: 1. 远程代理:代理对象可以代表一个位于远程系统...

    Android设计模式之代理模式(Proxy Pattern)

    代理模式是设计模式的一种,它的主要目的是在不改变原有对象的基础上,为一个对象提供额外的功能或者控制对这个对象的访问。在Android开发中,代理模式的应用尤为常见,尤其在处理复杂的业务逻辑、网络请求、界面...

    java 设计模式之代理模式(Proxy Pattern)实现代码及设计详解:动态代理模式、静态代理模式

    在这些模式中,代理模式(Proxy Pattern)是一种常用的结构型设计模式,它允许我们为一个对象创建一个代理,该代理对象在客户端和目标对象之间起到中介的作用,可以增加额外的功能或控制访问。 代理模式分为两种...

    Android设计模式之代理模式Proxy浅显易懂的详细说明

    代理模式也是平时比较常用的设计模式之一,代理模式其实就是提供了一个新的对象,实现了对真实对象的操作,或成为真实对象的替身.在日常生活中也是很常见的.例如A要租房,为了省麻烦A会去找中介,中介会替代A去筛选房子,A...

    Java24种设计模式,Java24种设计模式,24种设计模式,学会了这24种设计模式,可以打遍天下无敌手,设计模式非常重要

    2、代理模式PROXY PATTERN 3、单例模式SINGLETON PATTERN 4、多例模式MULTITION PATTERN 5、工厂方法模式FACTORY METHOD PATTERN 6、抽象工厂模式ABSTRACT FACTORY PATTERN 7、门面模式FACADE PATTERN 8、适配器...

    代理模式java代码 Proxy(4)

    代理模式是一种设计模式,它允许我们为一个对象创建一个代理对象,这个代理对象在客户端和目标对象之间起到中介的作用,可以实现额外的功能,比如监控、权限控制、事务管理等,而客户端无需关心这些细节。...

    Proxy代理模式经典实例(绝对OOP)

    Proxy代理模式是面向对象设计模式中的重要组成部分,它在软件工程中扮演着关键角色,尤其在Java编程中。代理模式的主要目的是为一个对象提供一个替身或者占位符,以便控制对这个对象的访问。这种设计模式允许我们...

    面向对象初学者必须掌握的几种设计模式

    观察者模式 Observer:Swing中的事件模型 工厂模式 Factory:在JDK中遍地都是,比如JDBC、JNDI等,是...Strategy:java.util.Comparator就用到了它 代理模式 Proxy:学习Spring IoC容器必须掌握的模式 <br>

    proxy.rar java三种代理模式源码

    这个"proxy.rar"压缩包中包含了这三种代理模式的源代码示例,以及对泛型的应用。 1. 静态代理: 静态代理是最基础的代理形式,它通过创建一个代理类来实现目标接口,并在代理类中调用目标对象的方法。这种方式需要...

    代理模式小例子

    在代理模式中,有三个关键角色:真实目标(Real Subject)、代理(Proxy)和客户端(Client)。真实目标是代理所代表的对象,它执行实际的工作。代理则扮演真实目标的替身,它持有对真实目标的引用,并且在客户端与...

Global site tag (gtag.js) - Google Analytics