欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring靜態(tài)代理和動態(tài)代理代碼詳解

 更新時間:2017年11月27日 10:37:54   作者:zhouyeqin  
這篇文章主要介紹了Spring靜態(tài)代理和動態(tài)代理代碼詳解,具有一定參考價值,需要的朋友可以了解下。

本節(jié)要點:

Java靜態(tài)代理
Jdk動態(tài)代理

1 面向?qū)ο笤O計思想遇到的問題

在傳統(tǒng)OOP編程里以對象為核心,并通過對象之間的協(xié)作來形成一個完整的軟件功能,由于對象可以繼承,因此我們可以把具有相同功能或相同特征的屬性抽象到一個層次分明的類結(jié)構(gòu)體系中。隨著軟件規(guī)范的不斷擴大,專業(yè)化分工越來越系列,以及OOP應用實踐的不斷增多,隨之也暴露了一些OOP無法很好解決的問題。

現(xiàn)在假設系統(tǒng)中有三段完全相似的代碼,這些代碼通常會采用“復制”、“粘貼”方式來完成,通過這種方式開發(fā)出來的軟件如圖所示:

可能讀者已經(jīng)發(fā)現(xiàn)了這種做法的不足之處,如果有一天,藍色背景的代碼需要修改,那是不是要同時修改三個地方?如果不僅僅是這三個地方包含這段代碼,而是100個,甚至是1000個地方,那會是什么后果?

記錄日志在代碼中無處不在---先來看一個例子:

為了跟蹤應用程序的運行過程,很多方法都需要記錄日志信息。我們一般這樣寫:

//log4j的使用見文章“l(fā)og4j介紹”
import org.apache.log4j.Logger;
public class Person {
	private Logger logger = Logger.getLogger(Person.class);
	public void sleep(){
		logger.info(“開始執(zhí)行時間:“ + new Date());
		System.out.println("睡覺中");
		logger.info(“執(zhí)行結(jié)束時間:” + new Date());
	}
	public void eating(){
		logger.info("開始執(zhí)行時間:“ + new Date()");
		System.out.println("正在吃飯中");
		logger.info("“執(zhí)行結(jié)束時間:” + new Date()");
	}
}

提問:弊端在哪里?

l混淆了業(yè)務方法本身的職責

l維護工作量巨大

2解決方案1

靜態(tài)代理:
   1、需要知道核心類(被代理類)是哪一個類,并且有什么方法?!?br />    2、非核心的代碼需要重復寫多次,顯得代碼的結(jié)構(gòu)臃腫,形成代碼冗余。
   3、非核心類(代理類)需要實現(xiàn)核心類(被代理類)實現(xiàn)的接口,也就是他們需要實現(xiàn)共同的接口,但是以核心類實現(xiàn)的接口(被代理類)為準。

l目地是將業(yè)務代碼與日志代碼完全分離,實現(xiàn)松散耦合.

l代理對象與被代理對象必須實現(xiàn)同一接口,在代理對象中實現(xiàn)與日志記錄的相關(guān)服務,并在需要的時候呼叫被代理對象,而被代理對象只保留業(yè)務代碼.

靜態(tài)代理的實現(xiàn)

1)定義接口:

public interface IPerson {
	public abstract void sleep();
	public abstract void eating();
}

2) 被代理類

public class Person implements IPerson {
	public void sleep(){
		System.out.println("睡覺中");
	}
	public void eating(){
		System.out.println("正在吃飯中");
	}
}

3) 代理類

import org.apache.log4j.Logger;
public class PersonProxy implements IPerson {
	private IPerson person;
	private Logger logger = Logger.getLogger(PersonProxy.class);
	public PersonProxy(IPerson person) {
		this.person = person;
	}
	public void eating() {
		logger.info("開始執(zhí)行時間:“ + new Date()");
		person.eating();
		logger.info("“執(zhí)行結(jié)束時間:” + new Date()");
	}
	public void sleep() {
		logger.info("開始執(zhí)行時間:“ + new Date()");
		person.sleep();
		logger.info("“執(zhí)行結(jié)束時間:” + new Date()");
	}
}

4) 測試類

package com.aptech.aop2;
public class PersonTest {
	public static void main(String[] args) {
		IPerson proxy = new PersonProxy(new Person());
		proxy.eating();
		proxy.sleep();
	}
}

靜態(tài)代理的弊端:

一個代理接口只能服務于一種類型的對象.對于稍大點的項目根本無法勝任.

3 解決方案2-動態(tài)代理

InvocationHandler:每一個動態(tài)代理類都必須實現(xiàn)InvocationHandler這個接口,并且每個代理類的實例都關(guān)聯(lián)到了一個handler,當我們通過代理對象調(diào)用一個方法的時候,這個方法的調(diào)用就會被轉(zhuǎn)發(fā)為由InvocationHandler這個接口的invoke方法來進行調(diào)用。

在JDK1.3之后加入了可協(xié)助開發(fā)的動態(tài)代理功能.不必為特定對象與方法編寫特定的代理對象,使用動態(tài)代理,可以使得一個處理者(Handler)服務于各個對象.
一個處理者的類設計必須實現(xiàn)java.lang.reflect.InvocationHandler接口.
通過InvocationHandler接口實現(xiàn)的動態(tài)代理只能代理接口的實現(xiàn)類.

動態(tài)代理實現(xiàn)

1) 處理者(Handler)

public class DynaProxyHandler implements InvocationHandler {
	private Logger logger = Logger.getLogger(DynaProxyHandler.class);
	private Object target;
	//被代理對象
	public void setTarget(Object target) {
		this.target = target;
	}
	public Object invoke(Object proxy, Method method, Object[] args)
	              throws Throwable {
		logger.info("執(zhí)行開始時間:" + new Date());
		Object result = method.invoke(target, args);
		logger.info("執(zhí)行結(jié)束時間:" + new Date());
		return result;
		//返回method執(zhí)行結(jié)果
	}
}

2) 生產(chǎn)代理對象的工廠

import java.lang.reflect.Proxy;
public class DynaProxyFactory {
	//obj為被代理對象
	public static Object getProxy(Object obj){
		DynaProxyHandler handler = new DynaProxyHandler();
		handler.setTarget(obj);
		return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), handler);
	}
}

3) 測試類

public class PersonTest {
	public static void main(String[] args) {
		IPerson person = (IPerson) DynaProxyFactory.getProxy(new Person());
		//返回代理類,代理類是JVM在內(nèi)存中動態(tài)創(chuàng)建的,該類實現(xiàn)傳入的接口數(shù)組的全部接口(的全部方法).
		person.eating();
		person.sleep();
	}
}

總結(jié)

以上就是本文關(guān)于Spring靜態(tài)代理和動態(tài)代理代碼詳解的全部內(nèi)容,希望對大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站:

Spring常用配置及解析類說明

SpringMVC攔截器實現(xiàn)單點登錄

Java編程實現(xiàn)springMVC簡單登錄實例

如有不足之處,歡迎留言指出。感謝朋友們對本站的支持!

相關(guān)文章

  • Java利用endorsed如何覆蓋jdk提供的類詳解

    Java利用endorsed如何覆蓋jdk提供的類詳解

    這篇文章主要給大家介紹了關(guān)于Java利用endorsed如何覆蓋jdk提供的類的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧。
    2017-09-09
  • IDEA之如何快速生成get和set方法

    IDEA之如何快速生成get和set方法

    這篇文章主要介紹了IDEA之如何快速生成get和set方法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • Java創(chuàng)建型設計模式之建造者模式詳解

    Java創(chuàng)建型設計模式之建造者模式詳解

    建造者模式是Java中一種創(chuàng)建型設計模式,它的主要目的是將一個復雜對象的構(gòu)建過程分解為多個簡單對象的構(gòu)建過程,本文將詳細介紹Java中的建造者模式,包括它的定義、結(jié)構(gòu)、實現(xiàn)方法以及應用場景等方面,希望對大家有所幫助
    2023-05-05
  • MyBatis批量插入數(shù)據(jù)到Oracle數(shù)據(jù)庫中的兩種方式(實例代碼)

    MyBatis批量插入數(shù)據(jù)到Oracle數(shù)據(jù)庫中的兩種方式(實例代碼)

    本文通過實例代碼給大家分享了MyBatis批量插入數(shù)據(jù)到Oracle數(shù)據(jù)庫中的兩種方式,非常不錯,具有參考借鑒價值,需要的朋友參考下吧
    2017-09-09
  • JPA save()方法將字段更新為null的解決方案

    JPA save()方法將字段更新為null的解決方案

    這篇文章主要介紹了JPA save()方法將字段更新為null的解決方案,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-01-01
  • jvm oom排查記錄剖析

    jvm oom排查記錄剖析

    這篇文章主要為大家介紹了jvm oom排查記錄剖析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-08-08
  • Intellij IDEA 2017新特性之Spring Boot相關(guān)特征介紹

    Intellij IDEA 2017新特性之Spring Boot相關(guān)特征介紹

    Intellij IDEA 2017.2.2版本針對Springboot設置了一些特性,本篇文章給大家簡單介紹一下如何使用這些特性,需要的朋友參考下吧
    2018-01-01
  • 如何利用IDEA快速生成實體類

    如何利用IDEA快速生成實體類

    這篇文章主要介紹了如何利用IDEA快速生成實體類問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • 深入分析@Resource和@Autowired注解區(qū)別

    深入分析@Resource和@Autowired注解區(qū)別

    這篇文章主要為大家介紹了深入分析@Resource和@Autowired注解區(qū)別,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-04-04
  • Spring Boot集成Kafka的示例代碼

    Spring Boot集成Kafka的示例代碼

    本篇文章主要介紹了Spring Boot集成Kafka的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-04-04

最新評論