`
guzizai2007
  • 浏览: 359582 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

代理模式

 
阅读更多

1、代理模式:

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

2、涉及的角色:

1)、抽象角色:声明真实对象和代理对象的共同接口。 
2)、代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象。同时,代理对象可以在执行真实对象操作时,附加其他的操作,相当于对真实对象进行封装。 
3)、真实角色:代理角色所代表的真实对象,是我们最终要引用的对象。

3、静态代理,抽象角色(Person):

package com.sxit.proxy;

public interface Person {
	
	//打招呼
	public void sayHello(String name);
	
	//吃东西
	public void eat(String food);
}

4、真实角色(Chinese实现Person接口): 

package com.sxit.proxy;

public class Chinese implements Person {
	
	public void sayHello(String name) {
		System.out.println("hi,"+name);
	}

	public void eat(String food) {
		System.out.println("I'm eat"+food);
	}
}

5、代理角色(Proxy): 

package com.sxit.proxy;

public class Proxy implements Person{

	//持有真实对象的引用,因为你要告诉这个代理类,你代理的是谁
	private Chinese chinese;
	
	//代理类构造函数初始化真实对象
	public Proxy(){
		if(chinese == null){
			chinese = new Chinese();
		}
	}

	//实现和真实对象一样的接口  执行真实对象业务逻辑之前、之后都可以插入一些其他的业务逻辑
	public void eat(String food) {
		preMethod();
		//调用真实对象eat方法
		chinese.eat(food);
		postMethod();
	}
	
	//实现和真实对象一样的接口  执行真实对象业务逻辑之前、之后都可以插入一些其他的业务逻辑
	public void sayHello(String name) {
		preMethod();
		//调用真实对象sayHello方法
		chinese.sayHello(name);
		postMethod();
	}
	
	public void preMethod(){
		System.out.println("方法调用之前------->>>>>>插入一段业务逻辑");
	}	
	
	public void postMethod(){
		System.out.println("方法调用之后------->>>>>>插入一段业务逻辑");
	}
}

6、测试类(StaticProxyTest):

package com.sxit.test;

import com.sxit.proxy.Proxy;

public class StaticProxyTest {

	public static void main(String[] args){
		
		//创建代理类  用户与代理直接交互,并不知道真实对象是谁,只通过代理类来执行自己需要的业务逻辑
		Proxy proxy = new Proxy();
		proxy.eat("香蕉");
		proxy.sayHello("傻逼");
	}
}

7、打印信息:

方法调用之前------->>>>>>插入一段业务逻辑
I'm eat香蕉
方法调用之后------->>>>>>插入一段业务逻辑
方法调用之前------->>>>>>插入一段业务逻辑
hi,傻逼
方法调用之后------->>>>>>插入一段业务逻辑

8、小结:

1)、以上代码可以看出,代理类持有真实对象的引用,用户只知道有代理类,而不知道真实对象是谁,当调用代理类的方法时,其实真正执行的是真实类中的业务逻辑,而代理类可以在前后穿插一些其他的业务逻辑。
2)、不足之处:因为这里的代理类持有一个真实对象的引用,真实对象是代理类中的一个属性,那等于是说一个实际对象就要对应一个代理类,大量使用会照成类数量膨胀,那肯定不合理,这是静态代理,那有没有更简单、方便的方式呢?

9、动态代理:

1)、Java动态代理类位于Java.lang.reflect包下,一般主要涉及到如下两个类
2)、Interface InvocationHandler:该接口中仅定义了一个方法Object:invoke(Object obj,Method method, Object[] args)。在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,如上例中的eat()、sayHello(),args为该方法的参数数组, 这个抽象方法在代理类中动态实现。 
3)、Proxy:该类即为动态代理类,作用类似于上例中的Proxy,其中主要包含以下内容:
Protected Proxy(InvocationHandler h)//构造函数
Static Class getProxyClass (ClassLoader loader, Class[] interfaces)//获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。 
Static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)//返回代理类的一个实例,返回后的代理类可以当作真实类使用(可使用真实类的在Person接口中声明过的方法)。 
4)、所谓Dynamic Proxy是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些 interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然啦,这个Dynamic Proxy其实就是一个Proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。

10、动态代理,接口角色(Person):

package com.sxit.proxy;

public interface Person {
	
	//打招呼
	public void sayHello(String name);
	
	//吃东西
	public void eat(String food);
}

11、具体角色:

package com.sxit.proxy;

public class Chinese implements Person {
	
	public void sayHello(String name) {
		System.out.println("hi,"+name);
	}

	public void eat(String food) {
		System.out.println("I'm eat"+food);
	}
}

12、 代理处理器:

package com.sxit.proxy;

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

//使用动态代理类时,我们必须实现InvocationHandler接口
public class ProxyHandler implements InvocationHandler{
	
	//代理的目标对象,不像静态代理强制指定某一类型对象
	private Object obj;
	
	//可接受任何类型对象为其代理
	public ProxyHandler(Object obj){
		this.obj = obj;
	}

	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

		preMethod();
		//obj是被代理的对象,args是被调用方法参数,返回值为真实对象调用该方法的返回值
		Object result = method.invoke(obj, args);
		postMethod();
		//如果该方法返回值 则为null
		return result;
	}
	
	public void preMethod(){
		System.out.println("方法调用之前------->>>>>>插入一段业务逻辑");
	}	
	
	public void postMethod(){
		System.out.println("方法调用之后------->>>>>>插入一段业务逻辑");
	}
}

13、测试类(DynamicProxyTest):

package com.sxit.test;

import com.sxit.proxy.Chinese;
import com.sxit.proxy.Person;
import com.sxit.proxy.ProxyHandler;
import java.lang.reflect.Proxy;

public class DynamicProxyTest {

	public static void main(String[] args) {
		
		Chinese chinese = new Chinese();
		ProxyHandler handler = new ProxyHandler(chinese);
		Person person = (Person)Proxy.newProxyInstance(chinese.getClass().getClassLoader(),chinese.getClass().getInterfaces(), handler);
		person.eat("香蕉");
		person.sayHello("傻逼");
	}
}

14、打印信息:

方法调用之前------->>>>>>插入一段业务逻辑
I'm eat香蕉
方法调用之后------->>>>>>插入一段业务逻辑
方法调用之前------->>>>>>插入一段业务逻辑
hi,傻逼
方法调用之后------->>>>>>插入一段业务逻辑

15、pre、post业务逻辑解耦,定义操作接口(IOperation):

package com.sxit.proxy;

public interface IOperation {

	//方法前操作
	public void preMethod(String name);
	
	//方法后操作
	public void postMethod(long id);
	
}

16、操作接口实现类(LoggerOperation): 

package com.sxit.proxy;

public class LoggerOperation implements IOperation {

	public void preMethod(String name) {
		System.out.println("方法调用之前------->>>>>>插入一段业务逻辑--------->>"+name);
	}

	public void postMethod(long id) {
		System.out.println("方法调用之后------->>>>>>插入一段业务逻辑--------->>"+id);
	}
}

 17、代理处理器(LogProxyHandler):

package com.sxit.proxy;

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

//使用动态代理类时,我们必须实现InvocationHandler接口
public class LogProxyHandler implements InvocationHandler{
	
	//代理的目标对象,不像静态代理强制指定某一类型对象
	private Object obj;
	
	//额外的操作对象 如LoggerOperation
	private Object proxy;
	
	//可接受任何类型对象为其代理
	public Object bind(Object obj,Object proxy){
		this.obj = obj;
		this.proxy = proxy;
		return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
	}

	/**
	 * 要处理的对象中的每个方法会被此方法送去JVM调用,也就是说,要处理的对象的方法只能通过此方法调用
     * 此方法是动态的,不是手动调用的
	 */
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

		//反射得到操作者类实例
		Class clazz = this.proxy.getClass();
		//反射得到操作者的preMethod方法
		Method preMethod = clazz.getMethod("preMethod", new Class[]{String.class});
		//反射执行preMethod方法
		preMethod.invoke(this.proxy, new Object[]{"日志中的姓名"});
		
		//执行要真实对象的原本方法 obj是被代理的对象,args是被调用方法参数,返回值为真实对象调用该方法的返回值
		Object result = method.invoke(obj, args);
		
		//反射得到操作者的postMethod方法
		Method postMethod = clazz.getMethod("postMethod", new Class[]{long.class});
		//反射执行preMethod方法
		postMethod.invoke(this.proxy, new Object[]{30});
		
		//如果该方法返回值 则为null
		return result;
	}
}

18、测试类(LogDynamicProxyTest): 

package com.sxit.test;

import com.sxit.proxy.Chinese;
import com.sxit.proxy.LogProxyHandler;
import com.sxit.proxy.LoggerOperation;
import com.sxit.proxy.Person;
import com.sxit.proxy.ProxyHandler;

public class LogDynamicProxyTest {

	public static void main(String[] args) {
		
		Chinese chinese = new Chinese();
		ProxyHandler handler = new ProxyHandler(chinese);
		Person person = (Person)new LogProxyHandler().bind(new Chinese(),new LoggerOperation());
		person.eat("香蕉");
		person.sayHello("傻逼");
	}
}

19、打印信息:

方法调用之前------->>>>>>插入一段业务逻辑--------->>日志中的姓名
I'm eat香蕉
方法调用之后------->>>>>>插入一段业务逻辑--------->>30
方法调用之前------->>>>>>插入一段业务逻辑--------->>日志中的姓名
hi,傻逼
方法调用之后------->>>>>>插入一段业务逻辑--------->>30

20、总结:

1)、现在真实对象也与操作类解耦,但是现在是所有方法都被日志记录,如果只希望eat()方法记录日志,而sayHello()方法不要,可以对传进来的method的名字进行判断,判断的条件存在XML里面.这样我们就可以配置文件时解藕了。  

 

      

分享到:
评论
发表评论

文章已被作者锁定,不允许评论。

相关推荐

    代理模式的使用示例程序

    代理模式是一种设计模式,它在软件工程中扮演着重要的角色,允许我们为其他对象提供一个替代接口,以控制对原始对象的访问。这种模式的主要目的是为了增加灵活性、安全性或者在不修改原有对象的情况下,增强或扩展其...

    JAVA设计模式之代理模式实例

    代理模式是设计模式的一种,它提供了一种对目标对象进行增强或者控制访问的方式。在本实例中,我们将深入探讨Java中的代理模式及其应用。 代理模式的核心思想是为一个对象创建一个代理对象,这个代理对象在客户端和...

    设计模式之代理模式Proxy

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

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

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

    android使用signalr代理模式和非代理模式

    SignalR提供了两种主要的工作模式:代理模式和非代理模式。这两种模式在实现上有所不同,各自具有优缺点,适用于不同的场景。 **1. 代理模式(Proxy Mode)** 在代理模式下,SignalR为每个Hub(服务端的业务逻辑...

    设计模式-代理模式

    代理模式是一种常用的设计模式,它在软件开发中扮演着重要的角色,特别是在iOS平台的应用程序设计中。代理模式的核心思想是为一个对象提供一个替身或代理,以控制对这个对象的访问。这种模式允许我们通过代理来间接...

    java中的三种代理模式

    在Java编程中,代理模式是一种常用的面向对象设计模式,它允许我们为一个对象提供一个代理以控制对该对象的访问。代理模式通常用于增加额外的功能,如日志、权限检查等,或者为了创建虚拟代理以提高性能。以下是Java...

    JAVA设计模式(代理模式)

    **Java设计模式——代理模式详解** 代理模式是软件设计模式中的一个重要组成部分,它在Java编程中扮演着举足轻重的角色。代理模式的核心思想是为一个对象提供一个替身,这个替身即代理对象,代理对象可以控制对原...

    代理模式小例子

    代理模式是一种设计模式,它在软件工程中扮演着重要的角色,允许我们为其他对象提供一个替代接口,以控制对原对象的访问。这种模式的主要目的是为了增加灵活性、安全性或者为对象提供额外的功能,同时保持客户端代码...

    设计模式之代理模式proxy

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

    设计模式实现——代理模式

    **设计模式实现——代理模式** 在软件工程中,设计模式是一种通用可重用的解决方案,它描述了在特定上下文中经常出现的问题以及该问题的解决方案。代理模式是设计模式的一种,它提供了一种对目标对象的间接访问方式...

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

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

    Java设计模式-代理模式例子

    在这个“Java设计模式-代理模式例子”中,我们将深入探讨代理模式的概念、实现方式以及它在实际开发中的应用。 代理模式的核心思想是为一个对象提供一个替身,这个替身即代理对象,代理对象控制对原对象的访问。在...

    android设计模式之代理模式

    代理模式在软件设计中是一种常用的设计模式,尤其在Android开发中,它可以帮助我们实现复杂的控制逻辑,隔离复杂性,以及提供额外的功能。在Android上下文中,代理模式常常用于数据加载、权限控制、事件处理等方面。...

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

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

    Java代理模式Java动态代理

    ### Java代理模式与Java动态代理详解 #### 一、代理模式概述 代理模式是一种软件设计模式,它在客户端和目标对象之间提供了一种间接层。这种模式的主要目的是控制客户端对目标对象的访问,并且可以在不修改原有...

    设计模式--代理模式

    代理模式是一种常用的设计模式,它在软件开发中扮演着重要角色,允许我们通过一个代理类来控制对原对象的访问。在《设计模式:可复用面向对象软件的基础》(通常称为GoF设计模式)中,代理模式被定义为“为其他对象...

    代理模式 C++实现

    代理模式(Proxy) 定义: 为其他对象提供一种代理以控制对这个对象的访问 结构: 由三部分组成 1.RealSubject(真实对象): 真正会调用到的对象 2.Proxy(代理对象): 代理真实对象的地方 3.Subject(共同点): 代理对象...

    cas代理模式代码示例

    在IT行业中,代理模式是一种常见的设计模式,它允许我们在不修改原有对象的基础上,为对象添加新的功能或控制访问。在本示例中,我们将重点讨论如何在Java环境下使用代理模式来实现代理逻辑,特别是在CAS(Central ...

Global site tag (gtag.js) - Google Analytics