亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java動態(tài)代理靜態(tài)代理實例分析

 更新時間:2020年03月16日 13:18:58   作者:譚寶貴  
這篇文章主要介紹了Java動態(tài)代理靜態(tài)代理實例分析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下

代理模式:為其他對象提供一種代理以控制某個對象的訪問。用在:在某些情況下,一個客戶不想或者不能直接訪問另一個對象,而代理對象可以在客戶端和目標(biāo)對象之前起到中介的作用,代理對象還可以完成它附加的操作。

例子:就像房東、租客、中介的關(guān)系。中介(代理對象)為房東(真實對象)出租房子,租客(客戶)通過中介(代理對象)來找房子租房子,中介完成了租房以后可以收取中介費(附加操作)。

先看看靜態(tài)代理模式,通過上面對代理模式的理解,可以了解到代理模式:即不直接通過new一個真實對象來調(diào)用方法,而是通過代理對象來調(diào)用一個方法,所以代理對象包含真實對象的引用。下面看一下代碼

接口:Subject包含一個方法

 package com.example.designpattern.proxy; 
 public interface Subject {
 void request();
 }

RealSubject類,實現(xiàn)了Subject接口,為了簡單起見,方法簡單的輸出一句話:

package com.example.designpattern.proxy;
public class RealSubject implements Subject {
//真是角色實現(xiàn)了
public void request() {
System.out.println("From real subject");
}
}

代理類ProxySubject,也要實現(xiàn)Subject接口,實現(xiàn)Subject里面的方法,但是在這里里面是通過調(diào)用真實對象來實現(xiàn)的。

package com.example.designpattern.proxy;

public class ProxySubject implements Subject {

private RealSubject realSubject; //代理角色內(nèi)部引用了真實角色

//代理角色實現(xiàn)目標(biāo)動作
public void request() {

this.preRequest(); //在真實角色操作之前所附加的操作
if (realSubject == null){
realSubject = new RealSubject();
}
realSubject.request(); // 真實角色所完成的事情
this.afterRequet(); //在真實角色操作之后附加的操作
}
//代理角色之前完成的動作
private void preRequest(){
System.out.println("pre request");
}
//代理角色之后完成的動作
private void afterRequet(){
System.out.println("after request");
}
}

客戶調(diào)用者

package com.example.designpattern.proxy;

public class Client {
public static void main(String[] args) {
ProxySubject proxy = new ProxySubject();
//通過代理對象來調(diào)用方法
proxy.request(); 
}
}

靜態(tài)代理:

可以運行一下這些代碼哦, 可以在Client類中看到,是通過代理ProxySubject的對象proxy來調(diào)用方法的,在代理類ProxySubject中,有一個真實對象的引用,在代理對象的中request()方法調(diào)用了真實對象的方法。這樣的模式叫做代理模式。

優(yōu)點是:

1. 代理模式能將代理對象與真實對象被調(diào)用的目標(biāo)對象分離。

2. 一定程度上降低了系統(tǒng)的耦合度,擴(kuò)展性好。

代理類中包含了對真實主題的引用,這樣做也有缺點:

1. 真實對象與代理類一一對應(yīng),增加真實類也要增加代理類,這樣做會快速的增加類的數(shù)量,使得系統(tǒng)變得復(fù)雜。

2. 設(shè)計代理以前真實主題必須事先存在,不太靈活。

采用動態(tài)代理可以解決以上問題,動態(tài)代理是相對于靜態(tài)代理來說的。

可能你也會說怎么樣實現(xiàn)動態(tài)創(chuàng)建實例,以前我們創(chuàng)建實例不都是通過new 的方式來實現(xiàn)的嗎?

Hello hi = new Hello();

那么動態(tài)創(chuàng)建實例是由Java提供的功能,就不需要我們?nèi)ew 對象,他已經(jīng)定義好了靜態(tài)方法Proxy.newProxyInstance(),只要傳入?yún)?shù)調(diào)用就可以。Java文檔里面有哦,如圖:

Java標(biāo)準(zhǔn)庫提供了一種動態(tài)代理(DynamicProxy)的機制:可以在運行期動態(tài)創(chuàng)建某個interface的實例。

參數(shù)解釋:

 Proxy.newProxyInstance(
 ClassLoader loader, // 傳入ClassLoader
 Class<?>[] interfaces, // 傳入要調(diào)用的接口的方法數(shù)組
 InvocationHandler h); //傳入InvocationHandler 的實例

下面看一下動態(tài)代理例子代碼:

Subject 接口

package design.dynamicproxy;
public interface Subject {
void request(String str);
}

RealSubject類 實現(xiàn) Subject 接口

package design.dynamicproxy;
public class RealSubject implements Subject {
@Override
public void request(String str) {
System.out.println("From Real Subject!" + " args:" + str );
}
}

動態(tài)代理類DynamicSubject 實現(xiàn)了InvocationHandler,重寫invoke()方法

package design.dynamicproxy;

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

/**
* 該代理類的內(nèi)部屬性時Object類型,實際使用時,使用該類的構(gòu)造方法傳遞一個對象
* 此外該類還實現(xiàn)了invoke() 方法,該方法中的method.invoke() 其實就是要調(diào)用被代理對象的要執(zhí)行的方法
* 方法參數(shù)是object,表示該方法從屬于object對象,通過動態(tài)代理類,我們可以在執(zhí)行真是對象的
* 方法前后可以加入一些額外的方法
*/
public class DynamicSubject implements InvocationHandler {

//引入的類型是Object的,可以隨便傳入任何一個對象
private Object object;

public DynamicSubject(Object object){
this.object = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("before calling:" + method);
//等價于realSubject的request() 方法,如果這里不調(diào)用的話,不會調(diào)用Method 對象中的方法
method.invoke(object, args);
System.out.println("after calling:" + method);
return null;
}
}

Client類

package design.dynamicproxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
public class Client {
	public static void main(String[] args) {
		RealSubject realSubject = new RealSubject();
		InvocationHandler handler = new DynamicSubject(realSubject);
		Class<?> classType = handler.getClass();
		//下面的代碼一次性生成代理
		// 動態(tài)生成了class com.sun.proxy.$Proxy0 的實例,
		Subject subject = (Subject) Proxy.newProxyInstance(classType.getClassLoader(), realSubject.getClass().getInterfaces(),handler);
		subject.request("eather");
		System.out.println(subject.getClass());
		//輸出class com.sun.proxy.$Proxy0,可以看到Proxy.newProxyInstance() 是系統(tǒng)自動生成的實例
	}
}

在Client中可以看到,我們這里調(diào)用方法的是 subject.request("eather"); 這個對象subject 不是通過new DynamicSubject()生成的,而是Java內(nèi)部寫好的方法在運行時動態(tài)生成對象;可能有人說

InvocationHandler handler = new DynamicSubject(realSubject);

這里不是通過new new DynamicSubject(realSubject); 生成了一個對象嗎? 是的,但是它是InvocationHandler 類型的,主要是傳遞一個InvocationHandler類型參數(shù)給Proxy.newProxyInstance(); 即最后一個參數(shù)。通過Client類的最后一句輸出可以看到它是 class com.sun.proxy.$Proxy0 ,這是Java運行時生成的。

解決了靜態(tài)代理的難題:1. 真實對象與代理類一一對應(yīng),增加真實類也要增加代理類,這樣做會快速的增加類的數(shù)量,使得系統(tǒng)變得復(fù)雜。 為什么這么說呢, 因為代理類引用的類型是Object的,可以隨便傳入任何一個對象,當(dāng)真實類增加時,代理類不用增加,new DynamicSubject(object); new的時候把要傳入的對象傳進(jìn)去即可。

下面是Proxy.newProxyInstance(ClassLoader loader,Class<?>[] interfaces, InvocationHandler h); 這個方法的源碼啦,可以看看,深入了解一下

public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h)
throws IllegalArgumentException
{
	Objects.requireNonNull(h);
	final Class<?>[] intfs = interfaces.clone();
	final SecurityManager sm = System.getSecurityManager();
	if (sm != null) {
		checkProxyAccess(Reflection.getCallerClass(), loader, intfs);
	}
	/*
* Look up or generate the designated proxy class.
生成一個代理類對象
*/
	Class<?> cl = getProxyClass0(loader, intfs);
	/*
* Invoke its constructor with the designated invocation handler.
使用指定的調(diào)用處理程序調(diào)用其構(gòu)造函數(shù)。就是使用InvocationHandler 實例調(diào)用【要調(diào)用方法的那個類】的構(gòu)造方法
*/
	try {
		if (sm != null) {
			checkNewProxyPermission(Reflection.getCallerClass(), cl);
		}
		final Constructor<?> cons = cl.getConstructor(constructorParams);
		final InvocationHandler ih = h;
		if (!Modifier.isPublic(cl.getModifiers())) {
			AccessController.doPrivileged(new PrivilegedAction<Void>() {
				public Void run() {
					cons.setAccessible(true);
					return null;
				}
			}
			);
		}
		return cons.newInstance(new Object[]{
			h
		}
		);
	}
	catch (IllegalAccessException|InstantiationException e) {
		throw new InternalError(e.toString(), e);
	}
	catch (InvocationTargetException e) {
		Throwable t = e.getCause();
		if (t instanceof RuntimeException) {
			throw (RuntimeException) t;
		} else {
			throw new InternalError(t.toString(), t);
		}
	}
	catch (NoSuchMethodException e) {
		throw new InternalError(e.toString(), e);
	}
}

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • SpringBoot實現(xiàn)連接nacos并支持多環(huán)境部署

    SpringBoot實現(xiàn)連接nacos并支持多環(huán)境部署

    這篇文章主要介紹了SpringBoot實現(xiàn)連接nacos并支持多環(huán)境部署方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-06-06
  • Spring Boot啟動過程(五)之Springboot內(nèi)嵌Tomcat對象的start教程詳解

    Spring Boot啟動過程(五)之Springboot內(nèi)嵌Tomcat對象的start教程詳解

    這篇文章主要介紹了Spring Boot啟動過程(五)之Springboot內(nèi)嵌Tomcat對象的start的相關(guān)資料,需要的朋友可以參考下
    2017-04-04
  • SpringBoot如何獲取application.properties中自定義的值

    SpringBoot如何獲取application.properties中自定義的值

    這篇文章主要介紹了SpringBoot獲取application.properties中的自定義的值,目錄結(jié)構(gòu)文件代碼給大家列舉的非常詳細(xì),需要的朋友可以參考下
    2021-09-09
  • 淺談@RequestMapping注解的注意點

    淺談@RequestMapping注解的注意點

    這篇文章主要介紹了淺談@RequestMapping注解的注意點,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • Java多維數(shù)組和Arrays類方法總結(jié)詳解

    Java多維數(shù)組和Arrays類方法總結(jié)詳解

    這篇文章主要介紹了Java多維數(shù)組和Arrays類方法總結(jié)詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-03-03
  • RocketMQ消息過濾與查詢的實現(xiàn)

    RocketMQ消息過濾與查詢的實現(xiàn)

    這篇文章主要介紹了RocketMQ消息過濾與查詢的實現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • Hibernate雙向多對多映射關(guān)系配置代碼實例

    Hibernate雙向多對多映射關(guān)系配置代碼實例

    這篇文章主要介紹了Hibernate雙向多對多映射關(guān)系配置代碼實例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-10-10
  • Spring Cloud下OAUTH2注銷的實現(xiàn)示例

    Spring Cloud下OAUTH2注銷的實現(xiàn)示例

    本篇文章主要介紹了Spring Cloud下OAUTH2注銷的實現(xiàn)示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-03-03
  • Java中documentHelper解析xml獲取想要的數(shù)據(jù)

    Java中documentHelper解析xml獲取想要的數(shù)據(jù)

    本文主要介紹了Java中documentHelper解析xml獲取想要的數(shù)據(jù),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-02-02
  • Java分層概念詳解

    Java分層概念詳解

    這篇文章主要介紹了Java分層概念詳解,內(nèi)容十分詳細(xì),在這里給大家分享下,需要的朋友可以參考。
    2017-09-09

最新評論