目录

Spring学习笔记

注解@service(“service”)括号中的service有什么用

service 是有用的相当于 xml配置中得bean id = service 也可以不指定 不指定相当于 bean id = com. service.service 就是这个类的全限定名,表示给当前类命名一个别名,方便注入到其他需要用到的类中;不加的话,默认别名就是当前类名,但是首字母小写

Spring注解@Component、@Repository、@Service、@Controller区别

所以,如果 Web 应用程序采用了经典的三层分层结构的话,最好在持久层、业务层和控制层分别采用 @Repository、@Service@Controller 对分层中的类进行注释,而用 @Component 对那些比较中立的类进行注释。

在一个稍大的项目中,通常会有上百个组件,如果这些组件采用xml的bean定义来配置,显然会增加配置文件的体积,查找以及维护起来也不太方便。 Spring2.5为我们引入了组件自动扫描机制,他可以在类路径底下寻找标注了@Component,@Service,@Controller,@Repository注解的类,并把这些类纳入进spring容器中管理。它的作用和在xml文件中使用bean节点配置组件时一样的。要使用自动扫描机制,我们需要打开以下配置信息: Java代码

1
2
3
4
5
6
7
8
9
<?xml version="1.0" encoding="UTF-8" ?> 
<beansxmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchemainstance"xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans 
http://www.springframework.org/schema/beans/spring-beans2.5.xsd  
http://www.springframework.org/schema/context  
http://www.springframework.org/schema/context/spring-context-2.5.xsd"  > 
 <context:component-scan base-package=”com.eric.spring”>   
 </beans>   

\6. 其中base-package为需要扫描的包(含所有子包) @Service用于标注业务层组件,@Controller用于标注控制层组件(如struts中的action),@Repository用于标注数据访问组件,即DAO组件,而@Component泛指组件,当组件不好归类的时候,我们可以使用这个注解进行标注。

1
2
3
4
5
6
@Service
public class VentorServiceImpl implements iVentorService{     
}   
Repository  
public class VentorDaoImpl implements iVentorDao {   
}   
  1. getBean的默认名称是类名(头字母小写),如果想自定义,可以@Service(“aaaaa”)这样来指定,这种bean默认是单例的,
  2. 如果想改变,可以使用
  3. @Service(“beanName”)
  4. @Scope(“prototype”)来改变。可以使用以下方式指定初始化方法和销毁方法(方法名任意):
1
2
3
4
5
6
7
@PostConstruct  
public void init() {   
 }   

 @PreDestroy  
 public void destory() {   
 }   

注入方式:

把DAO实现类注入到service实现类中,把service的接口(注意不要是service的实现类)注入到action中,注

入时不要new 这个注入的类,因为spring会自动注入,如果手动再new的话会出现错误,然后属性加上

@Autowired后不需要getter()和setter()方法,Spring也会自动注入。至于更具体的内容,等对注入的方式更

加熟练后会做个完整的例子上来。

注解:

在 spring的配置文件里面只需要加上<context:annotation-config/><context:component-scanbase-package="需要实现注入的类所在包"/>

可以使用base-package="*“表示全部的类。

< context:component-scan base-package=”com.eric.spring”>

其中base-package为需要扫描的包(含所有子包)

在接口前面标上@Autowired@Qualifier注释使得接口可以被容器注入,当接口存在两个实现类的时候必须指定其中一个来注入,

使用实现类首字母小写的字符串来注入,如:

1
2
3
@Autowired   
@Qualifier("chinese")    
private Man man;  

否则可以省略,只写@Autowired

@Service服务层组件,用于标注业务层组件,表示定义一个bean,自动根据bean的类名实例化一个首写字母为小写的bean,

例如Chinese实例化为chinese,如果需要自己改名字则:@Service("你自己改的bean名")

@Controller用于标注控制层组件(如struts中的action)

@Repository持久层组件,用于标注数据访问组件,即DAO组件

@Component泛指组件,当组件不好归类的时候,我们可以使用这个注解进行标注。

1
2
3
4
5
6
7
@Service 
public class VentorServiceImpl implements iVentorService { 
} 

@Repository 
public class VentorDaoImpl implements iVentorDao { 
} 

getBean 的默认名称是类名(头字母小写),如果想自定义,可以@Service(“aaaaa”) 这样来指定,这种

bean默认是单例的,如果想改变,可以使用@Service(“beanName”)``@Scope(“prototype”)来改变。

可以使用以下方式指定初始化方法和销毁方法(方法名任意):

1
2
3
4
5
6
7
@PostConstruct 
public void init() { 
} 

@PreDestroy 
public void destory() { 
} 

Spring中@Autowired注解、@Resource注解的区别

Spring不但支持自己定义的@Autowired注解,还支持几个由JSR-250规范定义的注解,它们分别是@Resource@PostConstruct以及@PreDestroy@Resource的作用相当于@Autowired,只不过@Autowired按byType自动注入,而@Resource默认按 byName自动注入罢了。@Resource有两个属性是比较重要的,分是name和type,Spring将@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型。所以如果使用name属性,则使用byName的自动注入策略,而使用type属性时则使用byType自动注入策略。如果既不指定name也不指定type属性,这时将通过反射机制使用byName自动注入策略。 @Resource装配顺序 \1. 如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常 \2. 如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常 \3. 如果指定了type,则从上下文中找到类型匹配的唯一bean进行装配,找不到或者找到多个,都会抛出异常 \4. 如果既没有指定name,又没有指定type,则自动按照byName方式进行装配;如果没有匹配,则回退为一个原始类型进行匹配,如果匹配则自动装配;

@Autowired@Resource的区别:

1、 @Autowired@Resource都可以用来装配bean.都可以写在字段上,或写在setter方法上。

2、 @Autowired默认按类型装配(这个注解是属业spring的),默认情况下必须要求依赖对象必须存在,如果要允许null值,可以设置它的required属性为false,如:@Autowired(required=false),如果我们想使用名称装配可以结合@Qualifier注解进行使用,如下:

1 @Autowired() @Qualifier("baseDao")
2 private BaseDao baseDao;

3、@Resource(这个注解属于J2EE的),默认安装名称进行装配,名称可以通过name属性进行指定,如果没有指定name属性,当注解写在字段上时,默认取字段名进行安装名称查找,如果注解写在setter方法上默认取属性名进行装配。当找不到与名称匹配的bean时才按照类型进行装配。但是需要注意的是,如果name属性一旦指定,就只会按照名称进行装配。

1 @Resource(name="baseDao")
2 private BaseDao baseDao;

推荐使用:@Resource注解在字段上,这样就不用写setter方法了,并且这个注解是属于J2EE的,减少了与spring的耦合。这样代码看起就比较优雅。

自定义Spring注解bean的命名策略

由于项目的需要spring的业务相关的bean不是写在xml文件中,因为项目是一个模块一个模块提交的,提交的时候不想修改xml文件,因此就用到了spring的注解Service。

例如:

Java代码

1
2
3
@Service("TestService")
public class TestService {
} 

这等同于:

1
<bean id="TestService" class="TestService"/>

spring会在classpath里面扫描标记有TestService等标签的类,扫描组件的配置如下:

Xml代码

1
2
<!-- 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代码

1
com.xyz.FooServiceImpl -> fooServiceImpl

观察spring的源代码发现,buildDefaultBeanName方法首先了bean名称小写的作用。

1
2
3
4
protected String buildDefaultBeanName (BeanDefinition definition){
    String shortClassName = ClassUtils.getShortName(definition.getBeanClassName());
    return Introspector.decapitalize(shortClassName);
}

因此,可以写一个类继承自AnnotationBeanNameGenerator,重写buildDefaultBeanName方法。

1
2
3
4
5
6
public class SdpAnnotationBeanNameGenerator extends AnnotationBeanNameGenerator {
    @Override
    protected String buildDefaultBeanName(BeanDefinitiondefinition) {
        return definition.getBeanClassName();
    }
}

我的改下代码:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
@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;
}

在扫描配置中需要添加自己的命名策略类:

1
2
<!-- sdp-service主要是注入服务类 -->
<context:component-scan base-package="org.sdp" name-generator="org.sdp.spring.SdpAnnotationBeanNameGenerator" />