原文: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元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
