Spring 父類變量注入失敗的解決
Spring 父類變量注入失敗
昨天遇到一個Action里面Service注入失敗,換種說法應(yīng)該說是根本沒有發(fā)生注入,本來很簡單的一個問題,但由于在項(xiàng)目中多個Action進(jìn)行了繼承,才最終導(dǎo)致了這個看似奇怪的問題。
下面小記下這個過程
收到同事問題,“有個Action請求一直調(diào)用報(bào)控指針,service一直是空的導(dǎo)致的!”
初步看了代碼及配置,沒有發(fā)現(xiàn)什么問題,起初懷疑是Action沒有g(shù)et方法所致,然后加上仍然無效;然后單步做了各種變量名的替換,一直一樣問題 ,這過程中一直關(guān)注java代碼確忽略了頁面請求,通過頁面請求發(fā)現(xiàn)代碼真正邏輯是頁面請求了一個子類Action的方法,而這個方法里面調(diào)用了父類的一個方法,此時父類里面的Service一直無法注入,對于上面所提的這種需求,實(shí)際上是需要在子類做Spring注入的同時也進(jìn)行父類的Spring注入,那么這種需要這樣的配置:
<bean id="****Action" class="com.**.**.contrl.**.mgr.action.**Action" scope="prototype" parent="termCommonAction"> <property name="orderVerifyApiFacade" ref="ord.bizprov.orderVerifyApiFacade"/> <property name="orderListQryApiFacade" ref="ord.query.orderListQryApiFacade"/> <property name="channelQryApiFacade" ref="cfguse.channel.channelQryApiFacade" /> </bean>
經(jīng)過上面的設(shè)置以后,請求子類的Action方法,子類方法中調(diào)用父類方法時,就不會出現(xiàn)父類不發(fā)生注入的問題了。
Spring通過父類注入公用屬性的技巧
XML配置方式提取父類
在使用Spring + Hibernate框架,或者SSH2等框架的時候,在開發(fā)中只有一個基本的DAO是現(xiàn)在的非常流行的做法。然后,在看過多份這種代碼以后,都是在每個業(yè)務(wù)類中聲明了一個DAO屬性,并且在Bean配置中,對每個業(yè)務(wù)類分別注入DAO。具體情形示例如下:
BaseDAO代碼:
public class BaseDAO {
public String service() {
return "Success!";
}
}
Services代碼:
//第一個業(yè)務(wù)類
public class ServiceA {
public String service() {
return baseDAO.service();
}
protected BaseDAO baseDAO;
public void setBaseDAO(BaseDAO baseDAO) {
this.baseDAO = baseDAO;
}
}
//第二個業(yè)務(wù)類
public class ServiceB {
public String service() {
return baseDAO.service();
}
protected BaseDAO baseDAO;
public void setBaseDAO(BaseDAO baseDAO) {
this.baseDAO = baseDAO;
}
}
Spring的Bean配置如下:
<bean id="baseDAO" class="com.watson.BaseDAO" /> <bean id="serviceA" class="com.watson.ServiceA"> <property name="baseDAO" ref="baseDAO" /> </bean> <bean id="serviceB" class="com.watson.ServiceB"> <property name="baseDAO" ref="baseDAO" /> </bean>
這樣的做法是現(xiàn)在的主流。這樣做不是說那里錯了,還是那句老話:這樣做肯定不優(yōu)美,誰讓人有時候是一根筋呢?
能夠想到的辦法是用一個父類來包含一些業(yè)務(wù)層公用的業(yè)務(wù)邏輯和屬性。所以可以將上面的代碼和配置。
Services代碼改寫如下:
//所有業(yè)務(wù)類的父類
public class BaseService {
protected BaseDAO baseDAO;
public void setBaseDAO(BaseDAO baseDAO) {
this.baseDAO = baseDAO;
}
}
//第一個業(yè)務(wù)類
public class ServiceA extends BaseService {
public String service() {
return baseDAO.service();
}
}
//第二個業(yè)務(wù)類
public class ServiceB extends BaseService {
public String service() {
return baseDAO.service();
}
}
Spring的Bean配置改寫如下:
<bean id="baseDAO" class="com.watson.BaseDAO" /> <bean id="BaseService" class="com.watson.BaseService" /> <property name="baseDAO" ref="baseDAO" /> </bean> <bean id="serviceA" class="com.watson.ServiceA" /> <bean id="serviceB" class="com.watson.ServiceB" />
這樣一來是不簡潔了很多?尤其在實(shí)際項(xiàng)目有太多Bean的時候。然后,這里不會達(dá)到我們預(yù)想的結(jié)果,因?yàn)檫@里會出現(xiàn)如下的錯誤:
exception:
org.springframework.web.util.NestedServletException: Request processing failed; nested exception is
java.lang.NullPointerException
......
root cause:
java.lang.NullPointerException:......
而出錯代碼就是每個業(yè)務(wù)中調(diào)用baseDAO的那行代碼。這說明注入失敗了。翻閱Spring的Bean注入詳解之后,很快就可以找應(yīng)該設(shè)置子類Bean配置的parent屬性。所以這里可以修改設(shè)置。
Spring的Bean配置改寫如下:
<bean id="baseDAO" class="com.watson.BaseDAO" /> <bean id="BaseService" class="com.watson.BaseService" /> <property name="baseDAO" ref="baseDAO" /> </bean> <bean id="serviceA" class="com.watson.ServiceA" parent="baseService" /> <bean id="serviceB" class="com.watson.ServiceB" parent="baseService" />
這個時候再運(yùn)行,就不會報(bào)錯了。原理是:在Spring的子類Bean配置中,其parent屬性作用是指定其父類,并繼承父類的注入屬性。不僅如此,子類還可以修改或者覆蓋父類的屬性值。例如上述代碼中的子類修改父類的baseDAO到屬性:
<bean id="BaseService" class="com.watson.BaseService" /> <property name="baseDAO" ref="baseDAO" /> </bean> <bean id="serviceA" class="com.watson.ServiceA" parent="baseService" /> <property name="baseDAO" ref="baseDAO2" /> </bean>
而對于父類的List等集合屬性,子類可以繼承父類的值,并且在其基礎(chǔ)上進(jìn)行增加新的值:
<bean id="BaseService" class="com.watson.BaseService" /> <property name="listValue"> <list> <value>listValue1</value> <value>listValue2</value> </list> </property> </bean> <bean id="serviceA" class="com.watson.ServiceA" parent="baseService" /> <property name="listValue"> <list> <value>listValue3</value> <value>listValue4</value> </list> </property> </bean>
Annotation方式提取父類
上面的方法是在XML配置文件中進(jìn)行的配置。而對現(xiàn)在Spring3流行的Annotation方式,其實(shí)更加的方便,完整示例如下:
BaseDAO代碼:
@Component
public class BaseDAO {
public String service() {
return "Success!";
}
}
Services代碼:
//所有業(yè)務(wù)類的父類
public class BaseService {
@Autowired
protected BaseDAO baseDAO;
}
//第一個業(yè)務(wù)類
@Component
public class ServiceA extends BaseService {
public String service() {
return baseDAO.service();
}
}
//第二個業(yè)務(wù)類
@Component
public class ServiceB extends BaseService {
public String service() {
return baseDAO.service();
}
}
Action層代碼:
@Controller
@RequestMapping(value = "/testaction")
public class TestAction {
@Autowired
private ServiceA service;
@RequestMapping(value = "/")
public @ResponseBody String home(Model model) {
return service.service();
}
}
這里根本就不需要進(jìn)行parent屬性子類的配置,可以完美的提取父類,并且可以順利的使用父類的公用屬性。至于原理,沒有去看源碼的處理方式,估計(jì)和上述XML配置是異曲同工的,只是在這里增加了對父類的檢測。
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringCloud對服務(wù)內(nèi)某個client進(jìn)行單獨(dú)配置的操作步驟
我們的微服務(wù)項(xiàng)目用的是springCloud,某個微服務(wù)接口因?yàn)閿?shù)據(jù)處理量大,出現(xiàn)了接口超時的情況,我們需要單獨(dú)修改這一個feignClient的超時時間,所以本文介紹了SpringCloud對服務(wù)內(nèi)某個client進(jìn)行單獨(dú)配置的操作步驟,需要的朋友可以參考下2023-10-10
Java異常中toString()和getMessage()區(qū)別
在java異常體系中,要打印異常信息,可以通過:e.getMessage() 、 e.toString() e.printStackTrace() 等方法打印,本文主要介紹了Java異常中toString()和getMessage()區(qū)別,具有一定的參考價值,感興趣的可以了解一下2024-01-01
SSH框架網(wǎng)上商城項(xiàng)目第5戰(zhàn)之商品類別級聯(lián)查詢和分頁功能
SSH框架網(wǎng)上商城項(xiàng)目第5戰(zhàn)之商品類別級聯(lián)查詢和分頁功能,寫一下CategoryServiceImpl實(shí)現(xiàn)類,完成數(shù)據(jù)庫的級聯(lián)查詢,感興趣的小伙伴們可以參考一下2016-05-05

