Spring注解@Component、@Repository、@Service、@Controller @Resource、@Autowired、@Qualifier、@scope

摘要:
当找不到一个匹配的Bean时,Spring容器将抛BeanCreationException异常,并指出必须至少拥有一个匹配的Bean。@Resource、@Autowired、@Qualifier当需要在某个类中定义一个属性,并且该属性是一个已存在的bean,要为该属性赋值我们就用着三个。@Autowired,它是org.springframework.beans.factory.annotation.Autowired是这个包中,它是spring的包。另外,@Autowired可以对成员变量、方法以及构造函数进行注释,而@Qualifier的标注对象是成员变量、方法入参、构造函数入参。

以下内容摘自部分网友的,并加上了自己的理解

@Service用于标注业务层组件(我们通常定义的service层就用这个)

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

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

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

这几个注解是当你需要定义某个类为一个bean,则在这个类的类名前一行使用@Service("XXX"),就相当于将这个类定义为一个bean,bean名称为XXX; 这几个是基于类的,我们可以定义名称,也可以不定义,不定义会默认以类名为bean的名称(类首字母小写)。

Spring 容器中匹配的候选 Bean 数目必须有且仅有一个。当找不到一个匹配的 Bean 时,Spring 容器将抛BeanCreationException 异常,并指出必须至少拥有一个匹配的 Bean。

@Resource、@Autowired、@Qualifier

当需要在某个类中定义一个属性,并且该属性是一个已存在的bean,要为该属性赋值我们就用着三个。例如:

<span style="font-size:18px;">@Resource
private IIocDao iocDao;
@Autowired
private IocService iocService;</span>

我们先看@Resource,它是javax.annotation.Resource; 这个包中,也就是说是javaEE中的,并不是spring中的

而且@Resource("xxx") 是可以定义bean名称的,就是说我这个属性要用那个bean来赋值。

@Autowired,它是org.springframework.beans.factory.annotation.Autowired 是这个包中,它是spring的包。

而且它没有@Autowired("xxx"),那我要为这个bean定义名称怎么办这个时候可以用@Qualifier("xxx") 这个也是spring中的。

另外,@Autowired 可以对成员变量、方法以及构造函数进行注释,而 @Qualifier 的标注对象是成员变量、方法入参、构造函数入参。

一般情况下Spring是通过类名来生成相应的bean的,如果一个接口有多个实现,那具体应该调用哪个bean呢?或者我不想使用和类名相同的bean名字怎么办呢?有以下两种

第一种:我们在生成bean的时候就给bean定义个名称

这样就把这个实现定义为myQuestionSysService了,而不是默认的类名questionSysService(生成的bean通常和类名相同,只是首字母小写)

//这里@Repository和使用@Service等是一样的

@Repository("myQuestionSysService")
public class QuestionSysImpl implements QuestionSysService {
	@Resource(name="questionSysDao")
	private QuestionSysDao  questionSysDao;
}

在使用这个bean的时候,使用

@Resource
private QuestionSysService myQuestionSysService;

第二种:在注入bean的时候指定名称

//在@Resource中指定name名字,和实现类的bean的名字相同

注意这里是定义的实现类还是使用的默认bean

<span style="font-size:18px;">@Resource(name = "</span><span style="font-size:18px;">q</span><span style="font-size:18px;">uestionSysImpl")
</span><pre name="code" class="java"><span style="font-size:18px;">private QuestionSysService xxx;</span>

如果你要为这个类指定别名bean,@Repository("myQuestionSysService"),那么@Resource(name="myQuestionSysService") 就要这么写了。就是这里的name要跟实现类对应的bean名称保持一致。而这里,private QuestionSysService xx; 这个属性名就随便写了。

如果用Autowired就要这么写了

@Autowired
@Qualifier("<span style="font-size:18px;">q</span><span style="font-size:18px;">uestionSysImpl</span>")
private QuestionSysService xx;

当然@Resoucre也可以像@Autowired这样指定@Qualifier,只不过@Resoucre可以直接指定bean name,而@Autowired不可以

记住一点:@Resource的作用相当于@Autowired,只不过@Autowired按byType自动注入,如果发现找到多个bean,则,又按照byName方式比对,如果还有多个,则报出异常 而@Resource默认按 byName自动注入罢了。其实spring注解,最常用的还是根据名称,根据类型啊,构造方法啊,用的非常少。所以在多个实现的时候我们定义好bean的名称就行,就不会错乱。

因为Autowired 不能像Resource 那样带个参数指定一个name,就要用Qualifier来指定了。

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

<?xml version="1.0" encoding="UTF-8"?>
<!--Spring总体配置-->
<beans xmlns="http://www.springframework.org/schema/beans"
		xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
		xmlns:context="http://www.springframework.org/schema/context"
		xmlns:tx="http://www.springframework.org/schema/tx"
		xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
				http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd
				http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
</beans>
				

最后是@scope用于指定bean在IOC容器中的生命周期

Spring最初只提供了两种scope,即singleton(默认的)和prototype,但在2.0之后,又发布了三种类型:request、session和global session,这三种只能在web应用中才能使用。

singleton: Spring 容器只会创建该bean定义的唯一实例,这个实例会被保存到缓存中,并且对该bean的所有后续请求和引用都将返回该缓存中的对象实例,一般情况下,无状态的bean使用该scope。

prototype:每次对该bean的请求都会创建一个新的实例,一般情况下,有状态的bean使用该scope。

request:每次http请求将会有各自的bean实例,类似于prototype。

session:在一个http session中,一个bean定义对应一个bean实例。

global session:在一个全局的http session中,一个bean定义对应一个bean实例。典型情况下,仅在使用portlet context的时候有效。

      无状态会话bean :bean一旦实例化就被加进会话池中,各个用户都可以共用。即使用户已经消亡,bean 的生命期也不一定结束,它可能依然存在于会话池中,供其他用户调用。由于没有特定的用户,那么也就不能保持某一用户的状态,所以叫无状态bean。但无状态会话bean 并非没有状态,如果它有自己的属性(变量),那么这些变量就会受到所有调用它的用户的影响,这是在实际应用中必须注意的。
      有状态会话bean :每个用户有自己特有的一个实例,在用户的生存期内,bean保持了用户的信息,即“有状态”;一旦用户灭亡(调用结束或实例结束),bean的生命期也告结束。即每个用户最初都会得到一个初始的bean。 

详见http://docs.spring.io/spring/docs/3.0.0.M3/reference/html/ch04s04.html

免责声明:文章转载自《Spring注解@Component、@Repository、@Service、@Controller @Resource、@Autowired、@Qualifier、@scope》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇JAVA安装查看oracle是否正常、表空间 (AIX)下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

在Spring中配置Hibernate事务

     本文主要探讨怎么用Spring来装配组件及其事务管理。在J2EE工程里连接到一个简单的数据库并不是什么难题,但是如果要综合组装企业类的组件就变得复杂了。一个简单的组件有一个或多个数据库支撑,所以,我们说到整合两个或多个的组件时,我们希望能够维持跨组件的许多数据库的运作的原子性。   J2EE提供了这些组件的容器,可以保证处理的原子性和独立性。在没...

Springboot2.x 启动报错:Bean named 'xxxService'... but was actually of type 'com.sun.proxy.$Proxy82'

Springboot 2.0.5 搭建一个新项目启动后报错:Bean named 'xxxService'... but was actually of type 'com.sun.proxy.$Proxy82' C:Javajdk1.8.0_144injava.exe -XX:TieredStopAtLevel=1 -noverify -Dspring....

Java bean 转为xml

Java bean 转为xml可以采用XStream类来完成 pom.xml <dependency> <groupId>com.thoughtworks.xstream</groupId> <artifactId>xstream</artifactId> <version>...

Dubbo多版本

当服务提供者提供的服务接口出现不兼容升级时,可以设置版本号,使用多个版本号(version)进行过渡。 1)、服务提供者配置文件 1 <dubbo:service ref="userService"interface="cn.coreqi.service.UserService"version="1.0.0"/> 2 3...

Spring 框架的事务管理

1. Spring 框架的事务管理相关的类和API PlateformTransactionManager 接口: 平台事务管理器(真正管理事务的类); TransactionDefinition 接口: 事务定义信息(事务的隔离级别,传播行为,超时,只读等); TransactionStatus 接口: 事务的状态; 平台事务管理器真正管理事务对...

三种方式创建bean对象在springIOC容器中初始化、销毁阶段要调用的自定义方法

1. 使用@Bean注解定义initMethod和destroyMethod 所谓initMethod和destroyMethod,是指在springIOC容器中,对于bean对象执行到初始化阶段和销毁阶段所调用的方法,其并不是初始化方法和销毁方法本身。 对于单例模式,initMethod会在创建容器时,构造方法、属性赋值方法完成之后调用,destroyM...