一.注解开发的意义
前面第四章所用的是XML配置的方式进行的开发,在实际的开发中,Action有多个,所对应的Service与Dao也有很多个,会导致存在很多的bean的情况。 而且,特别是简单的实体类POJO 会有很多的属性,会导致要写很多的<property> 进行注入,这样非常的不好,可以采用注解的方式进行相应的配置,可以简化开发。
例子与第四章的例子是一样的。
所使用的jar包有:
其中,spring-aop 与struts2-spring-plugin 的jar包是这次要用到的。其余的,都是上次第四章用到的。
二. 注解开发
添加注解,需要用到 aop的jar包。
二.一 添加aop 的jar 包。
二.二 在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>
二.八 重启服务器,看是否成功
可以正常的进行注入。
二.九 将action中@Autowired 注入到setter方法上
也是可以正常通过的。
三. 类注解
类上的注解有四个, @Component, @Controller,@Service, @Repository
以前只有一个注解 @Component, 现在扩充了后面的三个。 目前来说,这四个没有区别,但以后可能会扩充后面三个的意义,所以一般:
@Controller 用于Web层
@Service 用于事务层
@Repository 用于持久层
四. @Autowired 注解
@AutoWired 是按照类型,即byType 的方式进行装配的,默认是必须要存在的。
如果不存在的话,即在Action中注入service时,而将 service的注解进行去除:
这个时候,会报错的。 是在服务器启动的时候,就报错了。
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 时会报错,是空指向的异常。 服务器启动的时候不报错。
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
如果指定类的话:
@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 路径 下即可。
重启服务器,即可。 一定不要忘记引入整合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">
谢谢!!!