设计模式: 自己手动写一个代理模式

代理模式:为另一个对象提供一个替身或占位符以访问这个对象。代理模式为另一个对象提供代表,以便控制客户对对象的访问,管理访问的方式有许多种。

远程代理管理客户和远程对象之间的交互。

虚拟代理控制访问实例化开销大的对象。

保护代理基于调用者控制对对象方法的访问。

代理模式有许多变体,例如:缓存代理、同步代理、防火墙代理、写入时复制代理。

Java内置的代理支持,可以根据需要动态创建代理,并将所有调用分配到所选的调用处理器(InvocationHandler)。

下面 的例子是一种保护代理

类图:

源代码如下:

package javaproxy;
/**
 * 被代理者接口
 * @author Arvon
 *
 */
public interface PersonBean {
	public String getName();
	public String getGender();
	public String getInterests();
	public int getHotAndNotRating();

	public void setHotAndNotRating(int rating);
	public void setInterests(String interests);
	public void setName(String name);
	public void setGender(String gender);
}
package javaproxy;

public class PersonBeanImpl implements PersonBean {

	String name;
	String gender;
	String interests;
	int rating, ratingCount =1;

	public PersonBeanImpl(String name, String gender, String interests, int rating) {
		super();
		this.name = name;
		this.gender = gender;
		this.interests = interests;
		this.rating = rating;
	}

	@Override
	public String toString() {
		return "PersonBeanImpl [name=" + name + ", gender=" + gender + ", interests=" + interests + ", rating=" + getHotAndNotRating()
				+ "]";
	}

	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public String getGender() {
		return gender;
	}
	public void setGender(String gender) {
		this.gender = gender;
	}
	public String getInterests() {
		return interests;
	}
	public void setInterests(String interests) {
		this.interests = interests;
	}
	@Override
	public int getHotAndNotRating() {
		return ratingCount==0?0:(rating/ratingCount);
	}

	public void setHotAndNotRating(int rating){
		this.rating+=rating;
		ratingCount++;
	}

}
package javaproxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
/**
 * InvocationHandler 负责处理代理类发来的请求(即代理类的调用)
 * @author Administrator
 *
 */
public class OwnerInvocationHandler implements InvocationHandler {
	PersonBean person;
	public OwnerInvocationHandler(PersonBean person) {
		super();
		this.person = person;
	}
	/**
	 * Parameters:proxy - the proxy instance that the method was invoked
	 * onmethod - the Method instance corresponding to the interface method
	 * invoked on the proxy instance. The declaring class of the Method object
	 * will be the interface that the method was declared in, which may be a
	 * superinterface of the proxy interface that the proxy class inherits the
	 * method through.args - an array of objects containing the values of the
	 * arguments passed in the method invocation on the proxy instance, or null
	 * if interface method takes no arguments. Arguments of primitive types are
	 * wrapped in instances of the appropriate primitive wrapper class, such as
	 * java.lang.Integer or java.lang.Boolean.Returns:the value to return from
	 * the method invocation on the proxy instance. If the declared return type
	 * of the interface method is a primitive type, then the value returned by
	 * this method must be an instance of the corresponding primitive wrapper
	 * class; otherwise, it must be a type assignable to the declared return
	 * type. If the value returned by this method is null and the interface
	 * method's return type is primitive, then a NullPointerException will be
	 * thrown by the method invocation on the proxy instance. If the value
	 * returned by this method is otherwise not compatible with the interface
	 * method's declared return type as described above, a ClassCastException
	 * will be thrown by the method invocation on the proxy instance.
	 */
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws IllegalAccessException {
		try {
			if(method.getName().startsWith("get"))
				return method.invoke(person, args); //方法的执行者 person
			else if(method.getName().equals("setHotAndNotRating"))
				throw new IllegalAccessException();
			else if(method.getName().startsWith("set"))
				return method.invoke(person, args); //方法的执行者 person
		} catch (Exception e) {
//			e.printStackTrace();
			System.out.println("cannot set rating by owner proxy!");
		}
		return null;//如果调用其它的方法一律不予处理
	}

}
package javaproxy;

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

public class NonOwnerInvocationHandler implements  InvocationHandler{

	PersonBean person;
	public NonOwnerInvocationHandler(PersonBean person) {
		super();
		this.person = person;
	}
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		try {
			if(method.getName().startsWith("get"))
				return method.invoke(person, args); //方法的执行者 person
			else if(method.getName().equals("setHotAndNotRating"))
				return method.invoke(person, args);
			else if(method.getName().startsWith("set"))//他人不能设置自己的信息
				throw new IllegalAccessException();
		} catch (Exception e) {
			System.out.println("cannot set infos from nonowner proxy...");
		}
		return null;//如果调用其它的方法一律不予处理
	}
}
package javaproxy;

import java.lang.reflect.Proxy;
/**
 * 动态产生代理类
 * @author Administrator
 *
 */
public class ProxyHelper {
	/**
	 * Returns an instance of a proxy class for the specified interfaces that
	 * dispatches method invocations to the specified invocation handler.
	 * @param person
	 * @return
	 */
	public static PersonBean getOwnerProxy(PersonBean person) {
		return (PersonBean) Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(),
				new OwnerInvocationHandler(person));
	}
	public static PersonBean getNonOwnerProxy(PersonBean person) {
		return (PersonBean) Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(),
				new NonOwnerInvocationHandler(person));
	}

}
package javaproxy;
/**
 * 测试类
 * @author Administrator
 *
 */
public class ProxyTest {

	public static void main(String[] args) {
		PersonBean person = new PersonBeanImpl("Jay", "F", "Java", 7);
		System.out.println(person);
		 //动态地产生代理类ownerProxy (代理类ownerProxy实现了PersonBean接口)
		PersonBean ownerProxy = ProxyHelper.getOwnerProxy(person);
		System.out.println("Name is "+ownerProxy.getName());
		 //代理类的方法setInterests被调用  代理ownerProxy会把这个调用转发给OwnerInvocationHandler
		//ownerProxy 被实例化的时候指定的调用处理器是OwnerInvocationHandler
		// 接着 OwnerInvocationHandler 调用它自己的invoke方法
		//OwnerInvocationHandler 可能会转发给被代理者PersonBean person 处理,如 return method.invoke(person, args); //方法的执行者 person
		// 也可能做其他的处理 如抛出异常
		ownerProxy.setInterests("python");
		System.out.println(person);
		try {
			ownerProxy.setHotAndNotRating(8);
		} catch (Exception e) {
		}
		PersonBean nonOwnerProxy = ProxyHelper.getNonOwnerProxy(person);
		System.out.println("Name is "+nonOwnerProxy.getName());
		try {
			nonOwnerProxy.setHotAndNotRating(9);
			nonOwnerProxy.setInterests("python");
		} catch (Exception e) {
		}
		System.out.println(person);

	}

}

程序的输出:

PersonBeanImpl [name=Jay, gender=F, interests=Java, rating=7]

Name is Jay

PersonBeanImpl [name=Jay, gender=F, interests=python, rating=7]

cannot set rating by owner proxy!

Name is Jay

cannot set infos from nonowner proxy...

PersonBeanImpl [name=Jay, gender=F, interests=python, rating=8]

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

时间: 2024-10-10 04:12:04

设计模式: 自己手动写一个代理模式的相关文章

设计模式: 自己手动写一个状态模式

状态模式: 允许对象在内部状态改变时改变它的行为,对象看起来好像修改了它的类.状态模式将状态封装成独立的类,并将动作委托到代表当前状态的对象.状态模式使用组合通过简单引用不同的状态对象来造成类改变的假象. 状态模式和策略模式有相同的类图,但它们的意图不同,策略模式会用行为和算法来配置Context类.状态模式允许Context随着状态的改变而改变其行为. 源代码: </pre><pre name="code" class="java">pac

设计模式总结篇系列:代理模式(Proxy)

时代在发展,我们发现,现在不少明星都开始进行微访谈之类的,有越来越多的参与捐赠等.新的一天开始了,首先看下新的一天的日程安排: 1 interface Schedule{ 2 3 public void weiTalk(); 4 5 public void donation(); 6 7 } Schedule接口定义了今天的形成安排,主要包括微访谈和捐款.那么看一下实现此接口的明星类定义: 1 class Star implements Schedule { 2 3 @Override 4 pu

设计模式之第16章-代理模式(Java实现)

设计模式之第16章-代理模式(Java实现) “现在朋友圈真是太让人蛋疼了啊.”“怎么说?”“一堆代理,各种卖东西的,看着好烦人.”“哎,删了呗.”“都是朋友,哪里好意思删啊.”“这倒也是...哎,迫于生计,没办法咯.还好我不玩.”“对了,你不就是代理的鼻祖么,身为代理模式,你作何感想.”“以代理之道还治代理之身啊.” 代理模式之自我介绍 最近出场率超级高,哦不,一直以来出场率都挺高的说的大名鼎鼎的模式,就是我-代理模式是也.有关我的定义如下:Provide a surrogate or pla

java设计模式(10):代理模式(Proxy)

一,定义:  代理模式(Proxy):为其他对象提供一种代理以控制对这个对象的访问. 二,其类图: 三,分类一:静态代理 1,介绍:也就是需要我们为目标对象编写一个代理对象,在编译期就生成了这个代理对象,然后通过访问这个代理,来实现目标对象某些功能. 2,简单应用:在这里我们看一个简单的登录登出的例子: 登录功能的接口: [java] view plaincopyprint? /** * 公共接口,目标对象和代理都来实现 */ public interface ILogin{ //登录 void

.NET设计模式(14):代理模式(Proxy Pattern)(转)

摘要:在软件系统中,有些对象有时候由于跨越网络或者其他的障碍,而不能够或者不想直接访问另一个对象,如果直接访问会给系统带来不必要的复杂性,这时候可以在客户程序和目标对象之间增加一层中间层,让代理对象来代替目标对象打点一切.这就是本文要说的Proxy模式. 主要内容 1.例说Proxy模式 2.Proxy模式效果及实现要点 …… 概述 在软件系统中,有些对象有时候由于跨越网络或者其他的障碍,而不能够或者不想直接访问另一个对象,如果直接访问会给系统带来不必要的复杂性,这时候可以在客户程序和目标对象之

易学设计模式看书笔记(7) - 代理模式

代理模式 1.系统日志记录的例子:给系统中的业务逻辑加上日志 (1):最简单直接的做法 public class Test { private Logger logger = Loger.getLogger(this.getClass().getName()); public void doLgic(String name){ logger.log(name + "开始业务逻辑处理..."); //业务逻辑处理相关程序 System.out.println("业务逻辑处理相关

原来热加载如此简单,手动写一个 Java 热加载吧

1. 什么是热加载 热加载是指可以在不重启服务的情况下让更改的代码生效,热加载可以显著的提升开发以及调试的效率,它是基于 Java 的类加载器实现的,但是由于热加载的不安全性,一般不会用于正式的生产环境. 2. 热加载与热部署的区别 首先,不管是热加载还是热部署,都可以在不重启服务的情况下编译/部署项目,都是基于 Java 的类加载器实现的. 那么两者到底有什么区别呢? 在部署方式上: 热部署是在服务器运行时重新部署项目. 热加载是在运行时重新加载 class. 在实现原理上: 热部署是直接重新

面向对象的设计模式(十),代理模式

代理模式也称委托模式,是结构型设计模式之一,在实际应用中使用非常广泛,因此我们必须掌握这个设计模式. 定义:为一个对象提供一个代理对象,通过这个代理对象控制这个对象的行为. 使用场景: 直接访问或者控制对象的行为困难的时候,可以通过对象的代理对象间接控制对象的行为,为了使用简单透明,委托对象和代理对象需要实现相同的接口,即同类型,使用方法一样. 可以一定程度上保证对象的安全性.即不必直接将对象暴露给客户端,而是暴露该对象的代理对象给客户端,使得将该对象更加安全.但这不是主要的,不然就直接用原型模

菜鸟版JAVA设计模式—适配器模式,装饰模式,代理模式异同

节前把3个设计模式的学习心得分享了一下,分别是适配器模式,装饰模式,代理模式. 但是似乎越写越有些迷糊,这三种模式都有着一些大同小异的地方,总是有些搞不清. 今天再重新学习下,把学习心得分享给大家一下.这次是结合的我工作中的实际项目. 设计模式,其实是一种解决方法,JAVA作为一种面向对象的语言,对于所有的设计模式在实现上,总是少不了对接口的实现,对抽象类的继承,有时候似乎是为了去实现接口而去写接口,所以在这三种模式的说明代码中,我们都定义了接口这些东西,所以才有了现在的混淆. 先不厌其烦的介绍