您好,登錄后才能下訂單哦!
小編給大家分享一下Java注解機(jī)制之Spring自動(dòng)裝配實(shí)現(xiàn)原理的示例分析,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!
Java中使用注解的情況主要在SpringMVC(Spring Boot等),注解實(shí)際上相當(dāng)于一種標(biāo)記語言,它允許你在運(yùn)行時(shí)動(dòng)態(tài)地對(duì)擁有該標(biāo)記的成員進(jìn)行操作。注意:spring框架默認(rèn)不支持自動(dòng)裝配的,要想使用自動(dòng)裝配需要修改spring配置文件中<bean>標(biāo)簽的autowire屬性。
自動(dòng)裝配屬性有6個(gè)值可選,分別代表不同的含義:
byName ->從Spring環(huán)境中獲取目標(biāo)對(duì)象時(shí),目標(biāo)對(duì)象中的屬性會(huì)根據(jù)名稱在整個(gè)Spring環(huán)境中查找<bean>標(biāo)簽的id屬性值。如果有相同的,那么獲取這個(gè)對(duì)象,實(shí)現(xiàn)關(guān)聯(lián)。整個(gè)Spring環(huán)境:表示所有的spring配置文件中查找,那么id不能有重復(fù)的。
byType ->從Spring環(huán)境中獲取目標(biāo)對(duì)象時(shí),目標(biāo)對(duì)象中的屬性會(huì)根據(jù)類型在整個(gè)spring環(huán)境中查找<bean>標(biāo)簽的class屬性值。如果有相同的,那么獲取這個(gè)對(duì)象,實(shí)現(xiàn)關(guān)聯(lián)。
缺點(diǎn):如果存在多個(gè)相同類型的bean對(duì)象,會(huì)出錯(cuò);如果屬性為單一類型的數(shù)據(jù),那么查找到多個(gè)關(guān)聯(lián)對(duì)象會(huì)發(fā)生錯(cuò)誤;如果屬性為數(shù)組或集合(泛型)類型,那么查找到多個(gè)關(guān)聯(lián)對(duì)象不會(huì)發(fā)生異常。
constructor ->使用構(gòu)造方法完成對(duì)象注入,其實(shí)也是根據(jù)構(gòu)造方法的參數(shù)類型進(jìn)行對(duì)象查找,相當(dāng)于采用byType的方式。
autodetect ->自動(dòng)選擇:如果對(duì)象沒有無參數(shù)的構(gòu)造方法,那么自動(dòng)選擇constructor的自動(dòng)裝配方式進(jìn)行構(gòu)造注入。如果對(duì)象含有無參數(shù)的構(gòu)造方法,那么自動(dòng)選擇byType的自動(dòng)裝配方式進(jìn)行setter注入。
no ->不支持自動(dòng)裝配功能
default ->表示默認(rèn)采用上一級(jí)標(biāo)簽的自動(dòng)裝配的取值。如果存在多個(gè)配置文件的話,那么每一個(gè)配置文件的自動(dòng)裝配方式都是獨(dú)立的。
注解使用需要三個(gè)條件包括注解聲明,使用注解的元素,操作使用注解元素的代碼。第一步注解聲明,注解是一種類型,自定義注解編寫代碼如下:
package annotation; import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD) public @interface AttachAnnotation { String paramValue() default "河北省"; // 參數(shù)名為"paramValue" 默認(rèn)值為"河北省" }
使用自定義注解元素,代碼如下:
package annotation; /** * @author 路人宅 */ public class AttachEmlement { // 普通 public void AttachDefault(String name){ System.out.println("歸屬:" + name); } // 使用注解并傳入?yún)?shù) @AttachAnnotation(paramValue="河北省") public void AttachAnnotation(String name){ System.out.println("歸屬:" + name); } // 使用注解并使用默認(rèn)參數(shù) @AttachAnnotation public void AttachAnnotationDefault(String name){ System.out.println("歸屬:" + name); } }
測(cè)試操作執(zhí)行main函數(shù),具體代碼如下:
package annotation; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; public class AnnotionOperator { public static void main(String[] args) throws IllegalAccessException, IllegalArgumentException, InvocationTargetException, ClassNotFoundException { AttachEmlement element = new AttachEmlement(); // 初始化一個(gè)實(shí)例,用于方法調(diào)用 Method[] methods = AttachEmlement.class.getDeclaredMethods(); // 獲得所有方法 for (Method method : methods) { AttachAnnotation annotationTmp = null; if ((annotationTmp = method.getAnnotation(AttachAnnotation.class)) != null) method.invoke(element, annotationTmp.paramValue()); else method.invoke(element, "河南省"); } } }
執(zhí)行結(jié)果:
歸屬: 河南省
歸屬:河北省
歸屬:河北省
Spring為了方便自動(dòng)裝配進(jìn)行操作有兩種方式:繼承org.springframework.web.context.support.SpringBeanAutowiringSupport類或者添加@Component/@Controller等注解并在Spring配置文件里聲明context:component-scan元素配置。
1) 繼承方式實(shí)現(xiàn)自動(dòng)裝配,查看Spring3.1.1源代碼會(huì)發(fā)現(xiàn)SpringBeanAutowiringSupport類中有如下代碼:
/** * This constructor performs injection on this instance, * based on the current web application context. * Intended for use as a base class. * @see #processInjectionBasedOnCurrentContext */ public SpringBeanAutowiringSupport() { processInjectionBasedOnCurrentContext(this); }
分析:Java在實(shí)例化構(gòu)造時(shí)會(huì)調(diào)用默認(rèn)父類無參構(gòu)造方法,而Spring就是通過這一點(diǎn),讓操作元素代碼執(zhí)行的。
2) 通過注解方式的也和上述理論相似,值得注意的是注解自動(dòng)裝配無需完成注入setter*,查看Spring3.1.1源碼注解調(diào)用順序得出:
org.springframework.web.context.support.SpringBeanAutowiringSupport#SpringBeanAutowiringSupport=>
org.springframework.web.context.support.SpringBeanAutowiringSupport#processInjectionBasedOnCurrentContext=>
org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor#processInjection=>
org.springframework.beans.factory.annotation.InjectionMetadata#Injection,查看inject方法源代碼如下:
/** * Either this or {@link #getResourceToInject} needs to be overridden. */ protected void inject(Object target, String requestingBeanName, PropertyValues pvs) throws Throwable { if (this.isField) { Field field = (Field) this.member; ReflectionUtils.makeAccessible(field); field.set(target, getResourceToInject(target, requestingBeanName)); } else { if (checkPropertySkipping(pvs)) { return; } try { Method method = (Method) this.member; ReflectionUtils.makeAccessible(method); method.invoke(target, getResourceToInject(target, requestingBeanName)); } catch (InvocationTargetException ex) { throw ex.getTargetException(); } } }
分析:通過上述源碼Spring自動(dòng)裝配是通過反射機(jī)制來實(shí)現(xiàn)的。
看完了這篇文章,相信你對(duì)“Java注解機(jī)制之Spring自動(dòng)裝配實(shí)現(xiàn)原理的示例分析”有了一定的了解,如果想了解更多相關(guān)知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。