Spring的包扫描开发与@Autowired与@Resource注解的区别(五)

一.注解开发的意义


前面第四章所用的是XML配置的方式进行的开发,在实际的开发中,Action有多个,所对应的Service与Dao也有很多个,会导致存在很多的bean的情况。 而且,特别是简单的实体类POJO 会有很多的属性,会导致要写很多的<property> 进行注入,这样非常的不好,可以采用注解的方式进行相应的配置,可以简化开发。


例子与第四章的例子是一样的。


所使用的jar包有:


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


其中,spring-aop 与struts2-spring-plugin 的jar包是这次要用到的。其余的,都是上次第四章用到的。


二. 注解开发


添加注解,需要用到 aop的jar包。


二.一 添加aop 的jar 包。


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


二.二 在applicationContext.xml 中添加约束


添加的约束为 context。


添加后的约束为:


<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.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context.xsd">


二.三 添加组件扫描


其中,base-package 指定要扫描的包名, 如果有多个的话,之间用,进行隔开。


这个包名要尽量小,这样就可以少扫描一些无用的包,但要把


 <!-- 组件扫描 -->
    <context:component-scan base-package="com.yjl"></context:component-scan>
<!--多个的话-->
<context:component-scan base-package="com.yjl,com.abc,com.bcd"></context:component-scan>


二.四 在service中添加dao的引用


@Service
public class UserServiceImpl implements UserService{
    @Autowired
    private UserDao userDao;
    @Override
    public List<User> listAll() {
        System.out.println("输出userDao:"+userDao);
        return userDao.listAll();
    }
}


在UserServiceImpl 上,添加了一个@Service 的类注解, 在userDao 属性上面添加了一个@Autowired 的注解。 这里并没有实现userDao的setter和getter 方法。


二.五 UserDaoImpl


@Repository
public class UserDaoImpl implements UserDao {
    @Override
    public List<User> listAll() {
        List<User> userList=new ArrayList<User>();
        User user1=new User(1,"父亲","男",50);
        User user2=new User(2,"母亲","女",48);
        User user3=new User(3,"慧芳","女",27);
        User user4=new User(4,"正伟","男",28);
        User user5=new User(5,"莉莉","女",25);
        User user6=new User(6,"敬龙","男",26);
        User user7=new User(7,"两个蝴蝶飞","男",24);
        userList.add(user1);
        userList.add(user2);
        userList.add(user3);
        userList.add(user4);
        userList.add(user5);
        userList.add(user6);
        userList.add(user7);
        return userList;
    }
}


在UserDaoImpl 上添加了 @Repository 的注解, listAll() 的方法改成了普通的方法。


二.六 在Action 中注入service


@Controller
public class UserAction extends ActionSupport{
    private static final long serialVersionUID = 1L;
    @Autowired
    private UserService userService;
    /**
     * @author yuejl
     * @Description 查询全部
     * @return
     */
    public String list(){
        System.out.println("输出userService:"+userService);
        List<User> userList=userService.listAll();
        ServletActionContext.getRequest().setAttribute("userList",userList);
        return "list";
    }
    public UserService getUserService() {
        return userService;
    }
    public void setUserService(UserService userService) {
        this.userService = userService;
    }
}


在UserAction 上面有一个@Controller 的类注解, userService 属性上有一个@Autowired 的属性注解。


二.七 struts.xml 文件


<!--注意,此时class仍然为全限定名称-->
<action name="User_*" class="com.yjl.web.UserAction" method="{1}">
            <result name="list">list.jsp</result>
        </action>


二.八 重启服务器,看是否成功


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


可以正常的进行注入。


二.九 将action中@Autowired 注入到setter方法上


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


也是可以正常通过的。


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


三. 类注解


类上的注解有四个, @Component, @Controller,@Service, @Repository


以前只有一个注解 @Component, 现在扩充了后面的三个。 目前来说,这四个没有区别,但以后可能会扩充后面三个的意义,所以一般:


@Controller 用于Web层

@Service 用于事务层

@Repository 用于持久层


四. @Autowired 注解


@AutoWired 是按照类型,即byType 的方式进行装配的,默认是必须要存在的。


如果不存在的话,即在Action中注入service时,而将 service的注解进行去除:


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


这个时候,会报错的。 是在服务器启动的时候,就报错了。


Caused by: org.springframework.beans.factory.BeanCreationException: Could not autowire field:
 private com.yjl.service.UserService com.yjl.web.action.UserAction.userService; 
 nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: 
 No qualifying bean of type [com.yjl.service.UserService] found for dependency: 
 expected at least 1 bean which qualifies as autowire candidate for this dependency.
 Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}


可以手动设置 为非必须。


@Autowired(required=false)
    private UserService userService;


在访问 User_list.action 时会报错,是空指向的异常。 服务器启动的时候不报错。


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


2. 是按照类型进行的自动装配,现在 UserService 接口只有一个实现 UserServiceImpl 类,所以按照类型找的话,只会找到这一个,可以自动进行装配。 如果现在UserService 多了一个实现类型,即UserServiceImpl2时,这个时候会怎么样呢?


@Service
public class UserServiceImpl2 implements UserService{
    @Autowired
    private UserDao userDao;
    @Override
    public List<User> listAll() {
        System.out.println("输出userDao:"+userDao);
        return userDao.listAll();
    }
}


那么这个Action的话:


@Autowired
    private UserService userService;


这个时候会产生异常:


Caused by: org.springframework.beans.factory.BeanCreationException: Could not autowire field: 
private com.yjl.service.UserService com.yjl.web.action.UserAction.userService;
 nested exception is org.springframework.beans.factory.NoUniqueBeanDefinitionException: 
 No qualifying bean of type [com.yjl.service.UserService] is defined: 
expected single matching bean but found 2: userServiceImpl,userServiceImpl2


这个时候,如果按照类型的 话, 会找到两个实现, userServiceImpl 和userServiceImpl2 时,这个时候是没有办法区分的装配的,会出现错误。


3. 解决,针对上面的问题,可以添加另外一个注解 @Qualifier ,这个来指定名称。


如: @Qualifier(value="userServiceImpl") 这个userServiceImpl 表示的就是UserServiceImpl, @Qualifier(value="userServiceImpl2") 表示的就是UserServiceImpl2,


其中,有一个对应的关系, value 的名称为类名称首字母小写。


解决方式: 添加一个@Qualifier 的注解。


五 @Resource 注解


@Resource 的注解来源包是: import javax.annotation.Resource; 是java 规范的,而@Autowired来源包是: import org.springframework.beans.factory.annotation.Autowired;

故最好使用 @Resouce 的注解。 使用@Resource 需要指定一下名称。


五.一 UserDaoImpl


/**
 @author:yuejl
 @date: 2019年4月20日 上午10:17:09
 @Description 类的相关描述
*/
// 需要添加一个 value的值
@Repository(value="userDaoImpl")
public class UserDaoImpl implements UserDao {
    @Override
    public List<User> listAll() {
        List<User> userList=new ArrayList<User>();
        User user1=new User(1,"父亲","男",50);
        User user2=new User(2,"母亲","女",48);
        User user3=new User(3,"慧芳","女",27);
        User user4=new User(4,"正伟","男",28);
        User user5=new User(5,"莉莉","女",25);
        User user6=new User(6,"敬龙","男",26);
        User user7=new User(7,"两个蝴蝶飞","男",24);
        userList.add(user1);
        userList.add(user2);
        userList.add(user3);
        userList.add(user4);
        userList.add(user5);
        userList.add(user6);
        userList.add(user7);
        return userList;
    }

}


五.二 UserServiceImpl


//指定一个 value
@Service(value="userService")
public class UserServiceImpl implements UserService{
// 用name 指定是哪一个实现, 放置在属性上
    @Resource(name="userDaoImpl")
    private UserDao userDao;
    @Override
    public List<User> listAll() {
        System.out.println("输出userDao:"+userDao);
        return userDao.listAll();
    }
}


五.三 UserAction


@Controller
public class UserAction extends ActionSupport{
    private static final long serialVersionUID = 1L;
    private UserService userService;
    /**
     * @author yuejl
     * @Description 查询全部
     * @return
     */
    public String list(){
        System.out.println("输出userService:"+userService);
        List<User> userList=userService.listAll();
        ServletActionContext.getRequest().setAttribute("userList",userList);
        return "list";
    }
    
    public UserService getUserService() {
        return userService;
    }
    // 放置在setter方法上。
    @Resource(name="userService")
    public void setUserService(UserService userService) {
        
        this.userService = userService;
    }
}


其中 @Resource 的顺序 是:

引用于: https://blog.csdn.net/wingbin/article/details/38385269


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


如果指定类的话:


@Resource(name="userService",type=UserServiceImpl.class)


如果只有一个类型与其对应的话,可以直接即可。


@Resource
private UserService userService;


建议使用@Resouce


六. struts2-spring-plugin.jar


现在struts.xml 的配置依然是:


<action name="User_*" class="com.yjl.web.action.UserAction" method="{1}">
            <result name="list">list.jsp</result>
        </action>
    </package>


这个时候Dao,Service 的创建交给了Spring, 但是Action的创建并没有交给Spring, 仍然是Struts2 进行的创建。 虽然在UserAction上添加了@Controller 的注解。 这个时候,UserAction会创建两次, Spring通过注解@Controller 创建一次, Struts2通过class 全限定名称创建一次。 如果这里直接 class=“userAction” 是会出现错误的,因为 Spring 与Struts2 并没有相应的整合,两个是互不关联的。 需要将Struts2与Spring 进行整合,引入相应的jar 包,添加到classpath 路径 下即可。


Spring的包扫描开发与@Autowired与@Resource注解的区别(五)


重启服务器,即可。 一定不要忘记引入整合jar 包。


<package name="user" extends="struts-default" namespace="/">
    <action name="User_*" class="userAction" method="{1}">
            <result name="list">list.jsp</result>
        </action>
    </package>


Spring 创建的 实体bean 默认的是 类名称首字母小写的样式。 相当于xml配置的:


<bean id="userAction" class="com.yjl.web.action.UserAction">


谢谢!!!

上一篇:8行代码的web server


下一篇:我的mqtt协议和emqttd开源项目个人理解(14) - 使用redis插件来实现访问控制