原文:http://www.cnblogs.com/sunxucool/archive/2012/12/19/2824935.html
由于項目的需要spring的業務相關的bean不是寫在xml文件中,因為項目是一個模塊一個模塊提交的,提交的時候不想修改xml文件,因此就用到了spring的注解Service。
?例如:
? ? ??Java代碼
1 @Service("TestService")
2 public class TestService {
3 }
? ?這等同于:
1 <bean id="TestService" class="TestService"/>
? ?spring會在classpath里面掃描標記有TestService等標簽的類,掃描組件的配置如下:
??Xml代碼?
1 <!-- sdp-service主要是注入服務類 -->
2
3 <context:component-scanbase-package="org.sdp"/>
?加上以上的配置后,spring會自動的掃描org.sdp文件下的標記有注釋的類。
?以上的配置看似很“完美”,其實如果項目稍微大時就會出現問題,大家都知道spring的bean的id必須唯一,如果兩個人同事寫代碼就有可以造成寫同樣的bean名稱。
解決這個問題的一個思路是把bean的名稱修改為 類的全路徑,例如org.sdp.A 和com.bey.A 。
只要修改spring默認的bean命名策略就可以了。
AnnotationBeanNameGenerator是bean的默認命名策略,他實現了
BeanNameGenerator
接口。在Service里面,如果不寫bean的名稱,那么默認的名曾是類名,但是第一個字母是小寫的。
例如:
?Html代碼?
1 com.xyz.FooServiceImpl -> fooServiceImpl
觀察spring的源代碼發現,buildDefaultBeanName方法首先了bean名稱小寫的作用。
1 protected String buildDefaultBeanName(BeanDefinition definition) {
2 String shortClassName = ClassUtils.getShortName(definition.getBeanClassName());
3 return Introspector.decapitalize(shortClassName);
4
5 }
因此,可以寫一個類繼承自AnnotationBeanNameGenerator,重寫buildDefaultBeanName方法。
?Java代碼?
1 public class SdpAnnotationBeanNameGenerator extends AnnotationBeanNameGenerator {
2 @Override
3 protected String buildDefaultBeanName(BeanDefinitiondefinition) {
4 return definition.getBeanClassName();
5 }
6 }
我的改下代碼:
1 @Override
2 protected String buildDefaultBeanName(BeanDefinition definition) {
3 String className =definition.getBeanClassName();
4
5 className=className.substring(className.lastIndexOf(".")+1);
6 if(className.toLowerCase().endsWith("impl")){
7 className=className.substring(0, className.length()-4);
8 }
9 if((className.toLowerCase().endsWith("service")||className.toLowerCase().endsWith("dao"))==false){
10 return super.buildDefaultBeanName(definition);
11 }
12
13
14 className=className.substring(0,1).toLowerCase() + className.substring(1);
15 return className;
16 }
?
在掃描配置中需要添加自己的命名策略類:
1 <!-- sdp-service主要是注入服務類 -->
2 <context:component-scan base-package="org.sdp" name-generator="org.sdp.spring.SdpAnnotationBeanNameGenerator" />
通過以上配置,業務相關的bean不用寫bean的名稱了,ApplicationContext.getBean("類的全路徑")就可以得到類的實例了。
如果是spring2.5則就結束了,但是spring3.0為完美提供了getBean(name,requiredType);使用了泛型,因此只要傳入一個業務類的Class,getBean就返回此類的實例,而不用在
強制轉換類型了。
1 public static <T>T getService(String classFullName) throws ClassNotFoundException,BeansException{
2 Class<T> requiredType=(Class<T>)Class.forName(classFullName);
3 return SdpContext.getContext().getBean(classFullName,requiredType); }
強制轉換類型代碼:
??Java代碼?
1
TestService testService2=(TestService ) SdpContext.getContext().getBean("TestService "
);
2
System.out.println(testService2);
改進后的代碼:
?Java代碼?
1 @Service
2 public class TestService { }
1
TestService testService=SdpContext.getService("org.sdp.context.TestService"
);
2
System.out.println(testService);
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061
微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元

