spring注解(三)使用注解代替xml配置

来源:互联网 发布:yosemite mac 编辑:程序博客网 时间:2024/05/29 13:22

spring框架为我们提供了注解功能。

  使用注解编程,主要是为了替代xml文件,使开发更加快速。但是,xml文件的使用就是解决修改程序修改源代码,现在又不去使用xml文件,那么不就违背了开闭原则了么,得确是。不过么,注解也有注解的好,使用注解就不用配置那么多的xml文件啦,最重要的是开发效率高。。

  在没有使用注解时,spring框架的配置文件applicationContext.xml文件中需要配置很多的<bean>标签,用来声明类对象。使用注解,则不必在配置文件中添加标签拉,对应的是在对应类的“注释”位置添加说明。具体介绍如下:


使用注解前提

1,使用注解,需要在配置文件中增加命名空间和约束文件

[html] view plain copy
 print?
  1. <beans ...  
  2. xmlns:context="http://www.springframework.org/schema/context"  
  3. xsi:schemaLocation="  
  4. ...  
  5. http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context-2.5.xsd  
  6. ">  

  

2,告诉框架那些类是使用注解的。

<context:component-scan base-package="com.lsz.spring" />

项目中是在applicationContext.xml中



注解有哪些

spring框架使用的是分层的注解。

    持久层:@Repository

    服务层:@Service

    控制层:@Controller


持久层注解@Repository对应数据访问层Bean

[html] view plain copy
 print?
  1. Package com.lsz.spring;  
  2.    
  3. @Repository  
  4. public class UserDao{  
  5. //。。。。  
  6. }  

@Repository 等同于配置文件中的

[html] view plain copy
 print?
  1. <bean id="userDao" class="com.lsz.spring.UserDao" />  

当Service需要使用Spring创建的名字叫“userDao”的UserDaoImpl实例时,就可以使用@Resource(name = "userDao")注解告诉Spring,Spring把创建好的userDao注入给Service即可。

1 // 注入userDao,从数据库中根据用户Id取出指定用户时需要用到2 @Resource(name = "userDao")3 private BaseDao<User> userDao;


服务层注解@Service对应的是业务层Bean

[html] view plain copy
 print?
  1. @Service(value="testService")  
  2. public classTestService {  
  3.    
  4. @Resource//相当于自动装配  
  5. private UserDao userDao ;  
  6.    
  7.       public UserDao getUserDao() {  
  8.               returnuserDao;  
  9.       }  
  10.       public void setUserDao(UserDao userDao) {  
  11.              this.userDaouserDao;  
  12.       }  
  13.    
  14. }  
  15. @Resource 对象间关系的组合,默认采用的是byName方式进行装配,如果根据名称查找不到关联的对象,那么会再采用byType继续查找。

     

    @Service注解等同与

    [html] view plain copy
     print?
    1. <bean id="testService" class="com.lsz.spring.UserService" />  
在Action声明的“userService”变量的类型必须是“UserServiceImpl”或者是其父类“UserService”,否则由于类型不一致而无法注入,由于Action中的声明的“userService”变量使用了@Resource注解去标注,并且指明了其name = "userService",这就等于告诉Spring,说我Action要实例化一个“userService”,你Spring快点帮我实例化好,然后给我,当Spring看到userService变量上的@Resource的注解时,根据其指明的name属性可以知道,Action中需要用到一个UserServiceImpl的实例,此时Spring就会把自己创建好的名字叫做"userService"的UserServiceImpl的实例注入给Action中的“userService”变量,帮助Action完成userService的实例化,这样在Action中就不用通过“UserService userService = new UserServiceImpl();”这种最原始的方式去实例化userService了。如果没有Spring,那么当Action需要使用UserServiceImpl时,必须通过“UserService userService = new UserServiceImpl();”主动去创建实例对象,但使用了Spring之后,Action要使用UserServiceImpl时,就不用主动去创建UserServiceImpl的实例了,创建UserServiceImpl实例已经交给Spring来做了,Spring把创建好的UserServiceImpl实例给Action,Action拿到就可以直接用了。Action由原来的主动创建UserServiceImpl实例后就可以马上使用,变成了被动等待由Spring创建好UserServiceImpl实例之后再注入给Action,Action才能够使用。这说明Action对“UserServiceImpl”类的“控制权”已经被“反转”了,原来主动权在自己手上,自己要使用“UserServiceImpl”类的实例,自己主动去new一个出来马上就可以使用了,但现在自己不能主动去new“UserServiceImpl”类的实例,new“UserServiceImpl”类的实例的权力已经被Spring拿走了,只有Spring才能够new“UserServiceImpl”类的实例,而Action只能等Spring创建好“UserServiceImpl”类的实例后,再“恳求”Spring把创建好的“UserServiceImpl”类的实例给他,这样他才能够使用“UserServiceImpl”,这就是Spring核心思想“控制反转”,也叫“依赖注入”,“依赖注入”也很好理解,Action需要使用UserServiceImpl干活,那么就是对UserServiceImpl产生了依赖,Spring把Acion需要依赖的UserServiceImpl注入(也就是“给”)给Action,这就是所谓的“依赖注入”。对Action而言,Action依赖什么东西,就请求Spring注入给他,对Spring而言,Action需要什么,Spring就主动注入给他。


控制层注解@Controller对应表现层的Bean,也就是Action

[html] view plain copy
 print?
  1. @Controller(value="ua")  
  2. @Scope(value="prototype")  
  3. public class UserAction {  
  4.    
  5.      @Resource  
  6.      private UserService userService ;  
  7.    
  8.      public UserService getUserService() {  
  9.           returnuserService;  
  10.      }  
  11. }  

 @Controller注解等同于

[html] view plain copy
 print?
  1. <bean id="ua" class="com.lsz.spring.UserAction " />  

使用@Controller注解标识UserAction之后,就表示要把UserAction交给Spring容器管理,在Spring容器中会存在一个名字为"userAction"的action,这个名字是根据UserAction类名来取的。注意:如果@Controller不指定其value【@Controller】,则默认的bean名字为这个类的类名首字母小写,如果指定value【@Controller(value="UserAction")】或者【@Controller("UserAction")】,则使用value作为bean的名字

这里的UserAction还使用了@Scope注解,@Scope("prototype")表示将Action的范围声明为原型,可以利用容器的scope="prototype"来保证每一个请求有一个单独的Action来处理,避免struts中Action的线程安全问题。spring 默认scope 是单例模式(scope="singleton"),这样只会创建一个Action对象,每次访问都是同一Action对象,数据不安全,struts2 是要求每次次访问都对应不同的Action,scope="prototype" 可以保证当有请求的时候都创建一个Action对象



 这三个层中的注解关键字都可以使用@Component来代替。

 使用注解声明对象,默认情况下生成的id名称为类名称的首字母小写。

pring环境中获取Action对象。

[java] view plain copy
 print?
  1. ServletContext application =request.getSession().getServletContext();  
  2. ApplicationContextac = WebApplicationContextUtils.getWebApplicationContext(application);  
  3.    
  4. UserAction useraction = (UserAction)ac.getBean("ua");//获取控制层对象  
  5.   
  6. response.setContentType("text/html;charset=GBK");//设置编码  
  7. PrintWriter out =response.getWriter();  
  8.   
  9. //分别将三个层的对象打印出来。  
  10. out.println("Action:"+userAction);  
  11. out.println("Service:"+userAction.getUserService());  
  12. out.println("Dao:"+userAction.getUserService().getUserDao());  


阅读全文
0 0