设计模式之代理模式

本文详细介绍了代理模式的概念及其三种形式:简单代理、强制代理和动态代理。通过实例代码展示了每种代理模式的应用场景及其实现方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

代理模式定义:为其他对象提供一种代理以控制对这个对象的访问。

优点:

1、职责清晰。

2、高扩展性。

3、智能化。

下面分析三种形式的代理。

第一种、简单的代理模式

类图如下:


实现代码如下:

玩家抽象类:

package com.designpatterns.proxy;

public interface IGamePlayer {
	public void login(String name,String password);
	public void killBoss();
	public void upgrade();
}
玩家实现类:

package com.designpatterns.proxy;

public class GamePlayer implements IGamePlayer {
	private String name="";

	public GamePlayer(String name) {
		this.name=name;
	}

	@Override
	public void login(String name, String password) {
		System.out.println("用户名为:"+name+"的用户,"+this.name+"登录了!");

	}

	@Override
	public void killBoss() {
		System.out.println(this.name+"正在打怪!");

	}

	@Override
	public void upgrade() {
		System.out.println(this.name+"升一级!");

	}
}

代理实现类:

package com.designpatterns.proxy;

public class GamePlayerProxy implements IGamePlayer {

	private IGamePlayer gamePlayer=null;
	public GamePlayerProxy(IGamePlayer gamePlayer) {
		this.gamePlayer=gamePlayer;
	}

	@Override
	public void login(String name, String password) {
		this.gamePlayer.login(name, password);

	}

	@Override
	public void killBoss() {
		this.gamePlayer.killBoss();

	}

	@Override
	public void upgrade() {
		this.gamePlayer.upgrade();

	}
}

测试类:

package com.designpatterns.proxy;

public class Client {

	public static void main(String[] args) {
		IGamePlayer zs = new GamePlayer("张三");
		IGamePlayer proxy = new GamePlayerProxy(zs);
		proxy.login("zhangsan", "123456");
		proxy.killBoss();
		proxy.upgrade();

	}
}
第二种: 强制代理

这种情况下,必须通过制定的代理,才能访问到处理事件。

类图如下:


实现代码如下:

玩家抽象类;

package com.designpatterns.proxyforced;

public interface IGamePlayer {
	public void login(String name,String password);
	public void killBoss();
	public void upgrade();
	public IGamePlayer getProxy();
}

玩家实现类:

package com.designpatterns.proxyforced;

public class GamePlayer implements IGamePlayer {
	private String name="";
	private IGamePlayer proxy;

	public GamePlayer(String name) {
		this.name=name;
	}

	@Override
	public void login(String name, String password) {
		if (this.isProxy()) {
			System.out.println("用户名为:"+name+"的用户,"+this.name+"登录了!");
		}else {
			System.out.println("请使用制定的代理!");
		}
	}

	@Override
	public void killBoss() {
		if (this.isProxy()) {
			System.out.println(this.name+"正在打怪!");
		}else {
			System.out.println("请使用制定的代理!");
		}
	}

	@Override
	public void upgrade() {
		if (this.isProxy()) {
			System.out.println(this.name+"升一级!");
		}else {
			System.out.println("请使用制定的代理!");
		}
	}

	@Override
	public IGamePlayer getProxy() {
		if(this.proxy==null){
			this.proxy=new GamePlayerProxy(this);
		}
		return this.proxy;
	}

	private boolean isProxy(){
		if(this.proxy==null){
			return false;
		}else {
			return true;
		}
	}
}

代理实现类:

package com.designpatterns.proxyforced;

public class GamePlayerProxy implements IGamePlayer {

	private IGamePlayer gamePlayer=null;
	public GamePlayerProxy(IGamePlayer gamePlayer) {
		this.gamePlayer=gamePlayer;
	}

	@Override
	public void login(String name, String password) {
		this.gamePlayer.login(name, password);
	}

	@Override
	public void killBoss() {
		this.gamePlayer.killBoss();
	}

	@Override
	public void upgrade() {
		this.gamePlayer.upgrade();
	}

	@Override
	public IGamePlayer getProxy() {
		return this;
	}
}

测试类:

package com.designpatterns.proxyforced;

public class Client {
	public static void main(String[] args) {
		IGamePlayer zs = new GamePlayer("张三");
		System.out.println("-----------------------------\n错误的用法");
		IGamePlayer proxy = new GamePlayerProxy(zs);
		proxy.login("zhangsan", "123456");
		proxy.killBoss();
		proxy.upgrade();

		System.out.println("-----------------------------\n正确的用法");
		proxy = zs.getProxy();
		proxy.login("zhangsan", "123456");
		proxy.killBoss();
		proxy.upgrade();
	}
}

第三种、 动态代理

动态代理是在实现阶段不用关心代理谁,而在运行阶段才指定代理哪一个对象。(AOP就是典型的例子)

类图如下:


代码实现如下:

抽象的代理对象:

package com.designpatterns.dynamicproxy;

public interface Subject {
	public void sayHello();
	public void sayByeBye();
}

实现的代理对象;

package com.designpatterns.dynamicproxy;

public class RealSubject implements Subject {
	@Override
	public void sayHello() {
		System.out.println("hello!");
	}

	@Override
	public void sayByeBye() {
		System.out.println("ByeBye!");
	}
}

动态代理的Handler类:

package com.designpatterns.dynamicproxy;

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

public class MyInvocationHandler implements InvocationHandler {
	private Object target = null;

	public MyInvocationHandler(Object object) {
		this.target = object;
	}

	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		(new Before()).sayBefore();//AOP
		Object o = method.invoke(this.target, args);
		(new After()).sayAfter();//AOP
		return o;
	}
}

----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

在这里打断一下,为了体现AOP的编程思想,我们可以在这里实现一个简单的切面编程,我定义了两个方法。然后在动态代理的Handler类中调用。

After类:

package com.designpatterns.dynamicproxy;

public class After {
	public void sayAfter() {
		System.out.println("I am after!");
	}
}

Before类:

package com.designpatterns.dynamicproxy;

public class Before {
	public void sayBefore() {
		System.out.println("I am before!");
	}
}
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

动态代理类:

package com.designpatterns.dynamicproxy;

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

public class SubjectDynamicProxy<T> {
	@SuppressWarnings("unchecked")
	public static <T> T newProxyInstance(Object O) {
		ClassLoader loader = O.getClass().getClassLoader();
		Class<?>[] interfaces = O.getClass().getInterfaces();
		InvocationHandler h = new MyInvocationHandler(O);
		return (T) Proxy.newProxyInstance(loader, interfaces, h);
	}
}

测试类:

package com.designpatterns.dynamicproxy;


public class Client {

	public static void main(String[] args) {
		RealSubject realSubject= new RealSubject();
		Subject subjectProxy= SubjectDynamicProxy.newProxyInstance(realSubject);
		subjectProxy.sayHello();
		subjectProxy.sayByeBye();
		}
}

以上就是三种形式的代理模式。

参考资料
设计模式之禅

备注
转载请注明出处
http://blog.youkuaiyun.com/wsyw126/article/details/51313972
by WSYW126

代理模式是一种结构型设计模式,它提供一个代理对象来代表另一个对象。在代理模式中,有一个被称为实际对象(Subject)和一个被称为代理对象(Proxy)的中介,代理对象持有实际对象的引用,并且可以控制对实际对象的访问。代理模式的主要目的是在不修改原始对象的情况下,为原始对象添加额外的逻辑处理。 代理模式分为多种类型,如远程代理、虚拟代理、保护代理等,它们各自有不同的应用场景: - 远程代理:为远程对象提供一个本地代表。 - 虚拟代理:根据需要创建开销大的对象,通过虚拟代理控制访问这些对象的过程。 - 保护代理:控制对原始对象的访问权限,例如进行权限检查。 代理模式的优点包括: 1. 能够控制对真实对象的访问,并在访问前后添加额外的逻辑。 2. 可以通过代理对象实现延迟加载,即在实际需要时才创建真实对象。 3. 增强了对真实对象的封装,并且可以避免对真实对象的重复引用。 在C#中实现代理模式通常涉及以下步骤: 1. 定义一个接口或抽象类,声明真实对象和代理对象需要实现的方法。 2. 实现真实对象的类,按照接口或抽象类的要求实现具体方法。 3. 实现代理类,它同样实现接口或抽象类,并在方法中持有真实对象的引用,通过调用真实对象的方法来执行所需的操作,同时可以添加额外的逻辑。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值