搜尋
首頁Javajava教程Java 代理(Proxy)模式實作的程式碼詳解

類別圖


/**
 * 游戏者接口
 * @author stone
 * 
 */
public interface IGamePlayer {

	// 登录游戏
	public void login(String user, String password);

	// 杀怪,网络游戏的主要特色
	public void killBoss();

	// 升级
	public void upgrade();

}
/**
 * 游戏者
 * @author stone
 *
 */
public class GamePlayer implements IGamePlayer {

	private String name = "";

	// 通过构造函数传递名称
	public GamePlayer(String _name) {
		this.name = _name;
	}

	// 打怪,最期望的就是杀老怪

	public void killBoss() {

		System.out.println(this.name + " 在打怪!");

	}

	// 进游戏之前你肯定要登录吧,这是一个必要条件
	public void login(String user, String password) {
		System.out.println("登录名为" + user + " 的角色 " + this.name + "登录成功!");
	}

	// 升级,升级有很多方法,花钱买是一种,做任务也是一种
	public void upgrade() {
		System.out.println(this.name + " 又升了一级!");
	}

}


#
/**
 * 客户端 对被代理对象不可见
 */
public class GamePlayerProxy implements IGamePlayer {

	private IGamePlayer gamePlayer = null;//被代理对象

	// 通过构造函数传递要对谁进行代练
	public GamePlayerProxy(String username) {
		this.gamePlayer = new GamePlayer(username);
	}

	// 代练杀怪
	public void killBoss() {
		this.gamePlayer.killBoss();
	}

	// 代练登录
	public void login(String user, String password) {
		this.gamePlayer.login(user, password);
	}

	// 代练升级
	public void upgrade() {
		this.gamePlayer.upgrade();
	}

}


/*
 * 客户端 对被代理对象不可见
 */
public class GamePlayerProxy2 implements IGamePlayer {

	private IGamePlayer gamePlayer = null;//被代理对象

	// 通过构造函数传递要对谁进行代练
	public GamePlayerProxy2(String username) {
		this.gamePlayer = new GamePlayer(username);
	}

	// 代练杀怪
	public void killBoss() {
		this.gamePlayer.killBoss();
	}

	// 代练登录
	public void login(String user, String password) {
		System.out.println("登录时间是:" + new Date().toLocaleString());
		this.gamePlayer.login(user, password);
	}

	// 代练升级
	public void upgrade() {
		this.gamePlayer.upgrade();
		System.out.println("升级时间是:" + new Date().toLocaleString());
	}

}


#
/*
 * 客户端 对被代理对象不可见
 */
public class GamePlayerProxy3 {

	private IGamePlayer gamePlayer;
	// 通过构造函数传递 被代练(代理)对象
	public GamePlayerProxy3(IGamePlayer gamePlayer) {
		 this.gamePlayer = gamePlayer;
		 System.out.println("我是一名代练,我玩的角色是别人的,可以动态传递进来");
	}
	
	public IGamePlayer getProxy() {
		return (IGamePlayer) Proxy.newProxyInstance(this.getClass().getClassLoader(), 
				new Class[]{IGamePlayer.class}, new MyInvocationHandler());
	}

	private class MyInvocationHandler implements InvocationHandler {
	
		@Override
		public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
			if (method.getName().equals("login")) {
				System.out.println("登录时间是:" + new Date().toLocaleString());
			} if (method.getName().equals("upgrade")) {
				System.out.println("升级时间是:" + new Date().toLocaleString());
			}
			method.invoke(gamePlayer, args);
			return null;
		}
		
	}
}


/*
 * 代理模式:为其他对象提供一种代理以控制对这个对象的访问。
 * 		在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用
 * 优点
 (1).职责清晰
 真实的角色就是实现实际的业务逻辑,不用关心其他非本职责的事务,通过后期的代理完成一件完成事务,附带的结果就是编程简洁清晰。
 (2).代理对象可以在客户端和目标对象之间起到中介的作用,这样起到了的作用和保护了目标对象的作用。
 (3).高扩展性
 模式结构
 一个是真正的你要访问的对象(目标类),一个是代理对象,真正对象与代理
 对象实现同一个接口,先访问代理类再访问真正要访问的对象。
 
 在装饰器模式和代理模式之间还是有很多差别的。装饰器模式关注于在一个对象上动态的添加方法,然而代理模式关注于控制对对象的访问。
 换句话 说,用代理模式,代理类(proxy class)可以对它的客户隐藏一个对象的具体信息。因此,当使用代理模式的时候,我们常常在一个代理类中创建一个对象的实例。
 并且,当我们使用装饰器模 式的时候,我们通常的做法是将原始对象作为一个参数传给装饰者的构造器。
我们可以用另外一句话来总结这些差别:使用代理模式,代理和真实对象之间的的关系通常在编译时就已经确定了,而装饰者能够在运行时递归地被构造。    
 */
public class Test {
	public static void main(String[] args) {
		/*
		 * 普通的静态代理: 客户端不知道被代理对象,由代理对象完成其功能的调用
		 */
		IGamePlayer proxy = new GamePlayerProxy("X");
		System.out.println("开始时间是:" + new Date().toLocaleString());
		proxy.login("zhangSan", "abcd");
		proxy.killBoss();
		proxy.upgrade();
		System.out.println("结束时间是:" + new Date().toLocaleString());
		
		System.out.println();
		
		/*
		 * 代理对象 增强了 被代理对象的功能
		 */
		IGamePlayer proxy2 = new GamePlayerProxy2("M");
		proxy2.login("lisi", "efg");
		proxy2.killBoss();
		proxy2.upgrade();
		
		System.out.println();
		
		/*
		 * 动态代理:使用jdk提供的InvocationHandler,反射调用被代理对象的方法
		 * 	结合java.reflect.Proxy 产生代理对象
		 * 动态传入被代理对象构造InvocationHandler,在handler中的invoke时可以增强被代理对象的方法的功能
		 * 或者说:(面向切面:)在什么地方(连接点), 执行什么行为(通知)
		 * GamePlayerProxy3中是方法名为login时通知开始时间,upgrade时通知结束时间
		 */
		GamePlayerProxy3 dynamic = new GamePlayerProxy3(new GamePlayer("Y"));
		IGamePlayer dynamicPlayer = dynamic.getProxy();
		dynamicPlayer.login("wangwu", "1234");
		dynamicPlayer.killBoss();
		dynamicPlayer.upgrade();
		/*
		 * 面向切面: 一些相似的业务逻辑需要加在众多的地方,那们就可以把它提取到切面中, 切面也就是事务切面:如日志切面、权限切面、业务切面
		 */
	}
}

列印:


开始时间是:2014-10-8 17:19:05
登录名为zhangSan 的角色 X登录成功!
X 在打怪!
X 又升了一级!
结束时间是:2014-10-8 17:19:05

登录时间是:2014-10-8 17:19:05
登录名为lisi 的角色 M登录成功!
M 在打怪!
M 又升了一级!
升级时间是:2014-10-8 17:19:05

我是一名代练,我玩的角色是别人的,可以动态传递进来
登录时间是:2014-10-8 17:19:05
登录名为wangwu 的角色 Y登录成功!
Y 在打怪!
升级时间是:2014-10-8 17:19:05
Y 又升了一级!



以上是Java 代理(Proxy)模式實作的程式碼詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
JVM中的類加載程序子系統如何促進平台獨立性?JVM中的類加載程序子系統如何促進平台獨立性?Apr 23, 2025 am 12:14 AM

類加載器通過統一的類文件格式、動態加載、雙親委派模型和平台無關的字節碼,確保Java程序在不同平台上的一致性和兼容性,實現平台獨立性。

Java編譯器會產生特定於平台的代碼嗎?解釋。Java編譯器會產生特定於平台的代碼嗎?解釋。Apr 23, 2025 am 12:09 AM

Java編譯器生成的代碼是平台無關的,但最終執行的代碼是平台特定的。 1.Java源代碼編譯成平台無關的字節碼。 2.JVM將字節碼轉換為特定平台的機器碼,確保跨平台運行但性能可能不同。

JVM如何處理不同操作系統的多線程?JVM如何處理不同操作系統的多線程?Apr 23, 2025 am 12:07 AM

多線程在現代編程中重要,因為它能提高程序的響應性和資源利用率,並處理複雜的並發任務。 JVM通過線程映射、調度機制和同步鎖機制,在不同操作系統上確保多線程的一致性和高效性。

在Java的背景下,'平台獨立性”意味著什麼?在Java的背景下,'平台獨立性”意味著什麼?Apr 23, 2025 am 12:05 AM

Java的平台獨立性是指編寫的代碼可以在任何安裝了JVM的平台上運行,無需修改。 1)Java源代碼編譯成字節碼,2)字節碼由JVM解釋執行,3)JVM提供內存管理和垃圾回收功能,確保程序在不同操作系統上運行。

Java應用程序仍然可以遇到平台特定的錯誤或問題嗎?Java應用程序仍然可以遇到平台特定的錯誤或問題嗎?Apr 23, 2025 am 12:03 AM

Javaapplicationscanindeedencounterplatform-specificissuesdespitetheJVM'sabstraction.Reasonsinclude:1)Nativecodeandlibraries,2)Operatingsystemdifferences,3)JVMimplementationvariations,and4)Hardwaredependencies.Tomitigatethese,developersshould:1)Conduc

雲計算如何影響Java平台獨立性的重要性?雲計算如何影響Java平台獨立性的重要性?Apr 22, 2025 pm 07:05 PM

云计算显著提升了Java的平台独立性。1)Java代码编译为字节码,由JVM在不同操作系统上执行,确保跨平台运行。2)使用Docker和Kubernetes部署Java应用,提高可移植性和可扩展性。

Java的平台獨立性在廣泛採用中扮演著什麼角色?Java的平台獨立性在廣泛採用中扮演著什麼角色?Apr 22, 2025 pm 06:53 PM

Java'splatformindependenceallowsdeveloperstowritecodeonceandrunitonanydeviceorOSwithaJVM.Thisisachievedthroughcompilingtobytecode,whichtheJVMinterpretsorcompilesatruntime.ThisfeaturehassignificantlyboostedJava'sadoptionduetocross-platformdeployment,s

容器化技術(例如Docker)如何影響Java平台獨立性的重要性?容器化技術(例如Docker)如何影響Java平台獨立性的重要性?Apr 22, 2025 pm 06:49 PM

容器化技術如Docker增強而非替代Java的平台獨立性。 1)確保跨環境的一致性,2)管理依賴性,包括特定JVM版本,3)簡化部署過程,使Java應用更具適應性和易管理性。

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

EditPlus 中文破解版

EditPlus 中文破解版

體積小,語法高亮,不支援程式碼提示功能

ZendStudio 13.5.1 Mac

ZendStudio 13.5.1 Mac

強大的PHP整合開發環境

DVWA

DVWA

Damn Vulnerable Web App (DVWA) 是一個PHP/MySQL的Web應用程序,非常容易受到攻擊。它的主要目標是成為安全專業人員在合法環境中測試自己的技能和工具的輔助工具,幫助Web開發人員更好地理解保護網路應用程式的過程,並幫助教師/學生在課堂環境中教授/學習Web應用程式安全性。 DVWA的目標是透過簡單直接的介面練習一些最常見的Web漏洞,難度各不相同。請注意,該軟體中

MantisBT

MantisBT

Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

mPDF

mPDF

mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),