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

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节点配置组件时一样的。要使用自动扫描机制,我们需要打开以下

配置信息:

<?xml version="1.0" encoding="UTF-8" ?>
<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"
xsi:schemaLocation="http://www.springframework.org/schema/beans  
                    http://www.springframework.org/schema/beans/spring-beans-2.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>   
<!--其中base-package为需要扫描的包(含所有子包) @Service用于标注业务层组件,@Controller用于标注控制层组件
(如struts中的action),@Repository用于标注数据访问组件,即DAO组件,而@Component泛指组件,当组件不好归类的时候,
我们可以使用这个注解进行标注。-->

请看下面的一个例子;

dao层:

 package cn.lonecloud.pagekageauto.dao;
  /**
  * dao层的接口
  * @Title: UserDao.java
  * @Package cn.lonecloud.pagekageauto.dao
  */
 public interface UserDao {
     
     public void save();
}

dao层实现类:

package cn.lonecloud.pagekageauto.dao.impl;
 
 import org.springframework.stereotype.Repository;
 
 import cn.lonecloud.pagekageauto.dao.UserDao;
 /**
  * dao层的实现类
  * @Title: UserDaoImpl.java
  * @Package cn.lonecloud.pagekageauto.dao.impl
  */
 @Repository
 public class UserDaoImpl implements UserDao {
 
     @Override
     public void save() {
         System.out.println("测试保存");
     }
     //进行业务代码的实现
 }

Service层;

 package cn.lonecloud.pagekageauto.service;
 /**
  * service接口
  * @Title: UserService.java
  * @Package cn.lonecloud.pagekageauto.service
  */
 public interface UserService {
     public void Usersave();
 }

Service实现类:(@Service是标记实现类上的。因为@Service是把spring容器中的bean进行实例化,也就是等同于new操作,只有实现类是可以进行new实例化的,而接口则不能,所以是加在实现类上的。)

package cn.lonecloud.pagekageauto.service.Impl;
 
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.stereotype.Service;
 
 import cn.lonecloud.pagekageauto.dao.UserDao;
 import cn.lonecloud.pagekageauto.service.UserService;
 
 /**
  * service的实现类    
  * @Title: UserServiceImpl.java
  * @Package cn.lonecloud.pagekageauto.service.Impl
  */
 @Service
 public class UserServiceImpl implements UserService{
     //下面进行业务代码的实现
     @Autowired//自动注入
     UserDao userDao;
 
     @Override
     public void Usersave() {
         // TODO Auto-generated method stub
         System.out.println("service执行保存方法");
         userDao.save();
     }
     
 }

dto中间类:

 package cn.lonecloud.pagekageauto.dto;
 
 import org.springframework.stereotype.Component;
 /**
  * 中间类
  * @Title: DtoClass.java
  * @Package cn.lonecloud.pagekageauto.dto
  */
 @Component
 public class DtoClass {
     public void say(){
         System.out.println("我是dto");
     }
 }

控制层:

package cn.lonecloud.pagekageauto.Controller;
  
  import org.springframework.beans.factory.annotation.Autowired;
  import org.springframework.stereotype.Controller;
  
  import cn.lonecloud.pagekageauto.dto.DtoClass;
  import cn.lonecloud.pagekageauto.service.UserService;
  /**
  * 控制层类
  * @Title: UserController.java
  * @Package cn.lonecloud.pagekageauto.Controller
    */
 @Controller
 public class UserController {
     
     @Autowired
     private UserService userService;
     @Autowired
     public DtoClass dtoClass;
     /**
      * 保存方法
      * @Description:
      */
     public void save(){
         System.out.println("控制层save");
         dtoClass.say();
         userService.Usersave();
  }
 }
注解: 

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

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

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

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

在接口前面标上@Autowired和@Qualifier注释使得接口可以被容器注入,当接口存在两个实现类的时候必须指定其中一个来注入,使用实现类首字母小写的字符串来注入,如: 

    @Autowired      
    @Qualifier("chinese")       
    private Man man;    
否则可以省略,只写@Autowired  。 
@Service服务层组件,用于标注业务层组件,表示定义一个bean,自动根据bean的类名实例化一个首写字母为小写的bean,例如Chinese实例化为chinese,如果需要自己改名字则:@Service("你自己改的bean名")。   
@Controller用于标注控制层组件(如struts中的action) 
@Repository持久层组件,用于标注数据访问组件,即DAO组件 
@Component泛指组件,当组件不好归类的时候,我们可以使用这个注解进行标注。 

@Service 
public class VentorServiceImpl implements iVentorService { 

@Repository 
public class VentorDaoImpl implements iVentorDao { 

getBean 的默认名称是类名(头字母小写),如果想自定义,可以@Service(“aaaaa”) 这样来指定,这种 bean默认是单例的,如果想改变,可以使用@Service(“beanName”)@Scope(“prototype”)来改变。 

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

@PostConstruct 

public void init() { 



@PreDestroy 

public void destory() { 

}

@Autowired是根据类型进行自动装配的。如果当Spring上下文中存在不止一个UserDao类型的bean时,就会抛出BeanCreationException异常;如果Spring上下文中不存在UserDao类型的bean,也会抛出BeanCreationException异常。我们可以使用@Qualifier配合@Autowired来解决这些问题。如下:

①可能存在多个UserDao实例

  1. @Autowired
  2. @Qualifier( "userServiceImpl")
  3. public IUserService userService;

或者

  1. @Autowired
  2. public void setUserDao(@Qualifier("userDao") UserDao userDao) {
  3. this.userDao = userDao;
  4. }

这样Spring会找到id为userServiceImpl和userDao的bean进行装配。


②可能不存在UserDao实例

  1. @Autowired(required = false)
  2. public IUserService userService


 

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

Spring不但支持自己定义的@Autowired注解,还支持几个由JSR-250规范定义的注解,它们分别是@Resource@PostConstruct以及@PreDestroy
@Resource的作用相当于@Autowired,只不过@AutowiredbyType自动注入,而@Resource默认按 byName自动注入罢了。@Resource有两个属性是比较重要的,分是nametypeSpring@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型。所以如果使用name属性,则使用byName的自动注入策略,而使用type属性时则使用byType自动注入策略。如果既不指定name也不指定type属性,这时将通过反射机制使用byName自动注入策略。
@Resource装配顺序
1. 如果同时指定了nametype,则从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的耦合。这样代码看起就比较优雅


猜你喜欢

转载自blog.csdn.net/lzh_86/article/details/80921507
今日推荐