java設(shè)計模式-代理模式(實例講解)
代理模式是java最常見的設(shè)計模式之一。spring的aop就是使用了代理模式。
一般而言,代理模式分為靜態(tài)代理和動態(tài)代理兩種。
作為結(jié)構(gòu)類的設(shè)計模式,作用在于不修改類內(nèi)部代碼的情況下,對類進行拓展,是對繼承機制的一種補充。
eg :下面就用戶登錄這個例子實現(xiàn)一下代理模式。
基本需求是:實現(xiàn)用戶的登錄和修改昵稱功能。
上代碼,先是IUser接口和user實現(xiàn)類
public interface IUser {
//登錄
void login(String userId,String password);
//修改昵稱
void editNickname(String nickname);
}
public class User implements IUser {
private String nickname;
private String userId;
private String password;
public User(String userId,String password){
this.userId = userId;
this.password = password;
}
@Override
public void login(String userId, String password){
if(this.userId == userId && this.password == password){
System.out.println("用戶登錄成功");
}
else
System.out.println("用戶登錄失敗");
}
@Override
public void editNickname(String nickname) {
this.nickname = nickname;
System.out.println("修改昵稱成功,當前用戶的昵稱是:"+this.nickname);
}
}
客戶端類
public class Client {
public static void main(String[] args) {
//不調(diào)用代理模式時
IUser user = new User("firs","123");
user.login("firs", "123");
user.editNickname("大風");
}
還是非常簡單的??墒呛竺娈a(chǎn)品經(jīng)理跟你說,我們需要增加一個記錄用戶行為的功能,這下該怎么辦呢?直接修改user類?不不不,用代理模式。
增加一個代理類,在代理類里面寫“記錄用戶行為”的功能就好,不修改類,只拓展類,減少錯誤發(fā)生。
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
/**
* 靜態(tài)代理類必須實現(xiàn)接口,而且需要新創(chuàng)建一個類的代碼出來
* @author Administrator
*
*/
public class StaticProxy implements IUser {
private IUser user;
public StaticProxy(String userId,String password){
this.user = new User(userId,password);
}
//登陸前的操作,記錄當前登錄的時間
void noteLoginInfo(String[] params, String opreate){
Map<String,Object> loginInfo = new HashMap<>();
loginInfo.put("params", params);
loginInfo.put("opreate", opreate);
loginInfo.put("opreateTime", new Date());
System.out.println("記錄用戶操作成功");
}
@Override
public void login(String userId, String password){
noteLoginInfo(new String[]{userId, password},"login");
user.login(userId, password);
}
@Override
public void editNickname(String nickname) {
noteLoginInfo(new String[]{nickname},"editNickname");
user.editNickname(nickname);
}
}
客戶端類:
public class Client {
public static void main(String[] args) {
//不調(diào)用代理模式時
IUser user = new User("firs","123");
user.login("firs", "123");
user.editNickname("大風");
System.out.println("");
System.out.println("=============調(diào)用靜態(tài)代理模式后===========");
//需要實現(xiàn)記錄用戶登錄和修改昵稱操作的日志功能
//基于“拓展開發(fā),修改關(guān)閉”的設(shè)計準則,我們可以用靜態(tài)代理的方式
IUser proxy = new StaticProxy("firs","123");
proxy.login("firs", "123");
proxy.editNickname("我還是大風");
}
這樣子只需要修改客戶端類和增加靜態(tài)代理就可以了,完美實現(xiàn)??墒切枨笫菬o窮無盡的,產(chǎn)品經(jīng)理跟你說:“我們增加了一個管理員角色,還有二級管理員”啥啥啥的一大堆角色,
這就尷尬了,每個角色都要建一個靜態(tài)代理類,類爆炸了吧。不急,我們有動態(tài)代理模式。
動態(tài)代理模式在于不用自己新建代理類,你傳具體的實現(xiàn)類(主體)給他,他就默認給你生成了一個代理類。
從本質(zhì)上來說,它是利用了java的反射機制在運行時動態(tài)地生成了相應(yīng)的代理類。
沒有反射,就沒有動態(tài)代理。
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
/**
* 動態(tài)代理類不用和主體類繼承同一個接口
* @author Administrator
*
*/
public class DynamicProxy implements InvocationHandler {
private Object object;
public DynamicProxy(String userId,String password,Class<?> c){
Object obj = null;
try {
obj = Class.forName(c.getName())
.getConstructor(String.class,String.class)
.newInstance(userId,password);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
this.object = obj;
}
//登陸前的操作,記錄當前登錄的時間
void noteLoginInfo(String[] params, String opreate){
Map<String,Object> loginInfo = new HashMap<>();
loginInfo.put("params", params);
loginInfo.put("opreate", opreate);
loginInfo.put("opreateTime", new Date());
System.out.println("記錄用戶操作成功");
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
String[] params = new String[args.length];
for(int i = 0 ;i < args.length ; i++){
params[i] = args[i].toString();
}
noteLoginInfo(params, method.getName());
return method.invoke(object, args);
}
}
最后的客戶端類:
package com.test.my;
import java.lang.reflect.Proxy;
public class Client {
public static void main(String[] args) {
//不調(diào)用代理模式時
IUser user = new User("firs","123");
user.login("firs", "123");
user.editNickname("大風");
System.out.println("");
System.out.println("=============調(diào)用靜態(tài)代理模式后===========");
//需要實現(xiàn)記錄用戶登錄和修改昵稱操作的日志功能
//基于“拓展開發(fā),修改關(guān)閉”的設(shè)計準則,我們可以用靜態(tài)代理的方式
IUser proxy = new StaticProxy("firs","123");
proxy.login("firs", "123");
proxy.editNickname("我還是大風");
System.out.println("");
System.out.println("=============調(diào)用動態(tài)代理模式后===========");
DynamicProxy dynamicProxy = new DynamicProxy("firs","123",Admin.class);
ClassLoader cl = Admin.class.getClassLoader();
IUser iuser = (IUser)Proxy.newProxyInstance(cl,
new Class[]{IUser.class}, dynamicProxy);
iuser.login("firs","123");
iuser.editNickname("使用動態(tài)代理后的大風");
}
}
因為需求而增加的Admin類
public class Admin implements IUser {
private String nickname;
private String userId;
private String password;
public Admin(String userId,String password){
this.userId = userId;
this.password = password;
}
@Override
public void login(String userId, String password){
if(this.userId == userId && this.password == password){
System.out.println("用戶登錄成功");
}
else
System.out.println("用戶登錄失敗");
}
@Override
public void editNickname(String nickname) {
this.nickname = nickname;
System.out.println("修改昵稱成功,當前用戶的昵稱是:"+this.nickname);
}
}
總結(jié):
1.靜態(tài)代理模式相對來說比較簡單,要點在于對于每個實現(xiàn)類(subject主體)新建一個代理類,該代理類內(nèi)有實體類(subject主體)的引用,從而可以實現(xiàn)對原有實現(xiàn)類(subject主體)的控制,包括aop的控制等。
2.靜態(tài)代理是有局限性的,對于每個實體類可能都需要新建一個靜態(tài)代理類,這樣子可能會造成靜態(tài)代理類過多的情況,所以動態(tài)代理應(yīng)運而生了。
3.動態(tài)代理不局限于具體的實現(xiàn)類(subject主體),在其內(nèi)部是用object存取實體類的引用,再利用反射獲得該實體類的各種方法,從而實現(xiàn)對實現(xiàn)類(subject主體)的面向 切面AOP編程控制。
4.上述的寫法是JDK里的動態(tài)代理,不是特別完美,因為這種動態(tài)代理需要實體類實現(xiàn)至少一個接口。問題是并不是所有的類都會有接口,所以說不完美在這里。
上面都是我自己對于代理模式的理解,如有錯漏,還請批評指正,多謝。
以上這篇java設(shè)計模式-代理模式(實例講解)就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
- 詳解JAVA設(shè)計模式之代理模式
- Java設(shè)計模式之動態(tài)代理模式實例分析
- Java設(shè)計模式之靜態(tài)代理模式實例分析
- 23種設(shè)計模式(7) java代理模式
- java設(shè)計模式筆記之代理模式
- java設(shè)計模式學習之代理模式
- Java設(shè)計模式之代理模式與裝飾模式實例詳解
- Java設(shè)計模式之代理模式_動力節(jié)點Java學院整理
- java設(shè)計模式之代理模式(Porxy)詳解
- java設(shè)計模式—靜態(tài)代理模式(聚合與繼承方式對比)
- 詳解設(shè)計模式中的proxy代理模式及在Java程序中的實現(xiàn)
- Java使用設(shè)計模式中的代理模式構(gòu)建項目的實例展示
- 實例講解Java設(shè)計模式編程中如何運用代理模式
- Java設(shè)計模式之代理模式(Proxy模式)介紹
- Java通俗易懂系列設(shè)計模式之代理模式
相關(guān)文章
Java并發(fā)系列之CountDownLatch源碼分析
這篇文章主要為大家詳細介紹了Java并發(fā)系列之CountDownLatch源碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-03-03
使用ScheduledThreadPoolExecutor踩過最痛的坑
這篇文章主要介紹了使用ScheduledThreadPoolExecutor踩過最痛的坑及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-08-08
Java輕量級權(quán)限認證框架Sa-Token的使用
Sa-Token是一個輕量級Java權(quán)限認證框架,本文就詳細的來介紹一下Java輕量級權(quán)限認證框架Sa-Token的使用,主要解決:登錄認證、權(quán)限認證、Session會話、單點登錄、OAuth2.0、微服務(wù)網(wǎng)關(guān)鑒權(quán)等,感興趣的可以了解一下2022-03-03
詳解java封裝返回結(jié)果與RestControllerAdvice注解
這篇文章主要為大家介紹了java封裝返回結(jié)果與RestControllerAdvice注解實例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-09-09

