例如:
Java代码
@Service("TestService") public class TestService { }
这等同于:
<bean id="TestService" class="TestService"/>
spring会在classpath里面扫描标记有TestService等标签的类,扫描组件的配置如下:
Xml代码
<!-- sdp-service主要是注入服务类 --> <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代码
com.xyz.FooServiceImpl -> fooServiceImpl
观察spring的源代码发现,buildDefaultBeanName方法首先了bean名称小写的作用。
protected String buildDefaultBeanName(BeanDefinition definition) { String shortClassName = ClassUtils.getShortName(definition.getBeanClassName()); return Introspector.decapitalize(shortClassName); }
因此,可以写一个类继承自AnnotationBeanNameGenerator,重写buildDefaultBeanName方法。
public class SdpAnnotationBeanNameGenerator extends AnnotationBeanNameGenerator { @Override protected String buildDefaultBeanName(BeanDefinitiondefinition) { return definition.getBeanClassName(); } }
我的改下代码:
@Override protected String buildDefaultBeanName(BeanDefinition definition) { String className =definition.getBeanClassName(); className=className.substring(className.lastIndexOf(".")+1); if(className.toLowerCase().endsWith("impl")){ className=className.substring(0, className.length()-4); } if((className.toLowerCase().endsWith("service")||className.toLowerCase().endsWith("dao"))==false){ return super.buildDefaultBeanName(definition); } className=className.substring(0,1).toLowerCase() + className.substring(1); return className; }
在扫描配置中需要添加自己的命名策略类:
<!-- sdp-service主要是注入服务类 --> <context:component-scan base-package="org.sdp" name-generator="org.sdp.spring.SdpAnnotationBeanNameGenerator" />
参考:http://www.cnblogs.com/sunxucool/archive/2012/12/19/2824935.html