您的位置:首页 > 编程语言 > Java开发

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

2017-12-06 10:19 525 查看
spring框架为我们提供了注解功能。

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

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

使用注解前提

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

[html] view
plain copy

 print?

<beans ...  

xmlns:context="http://www.springframework.org/schema/context"  

xsi:schemaLocation="  

...  
http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context-2.5.xsd  
">  

  

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

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

项目中是在applicationContext.xml中

注解有哪些

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

    持久层:@Repository;

    服务层:@Service

    控制层:@Controller

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

[html] view
plain copy

 print?

Package com.lsz.spring;  

   

@Repository  

public class UserDao{  

//。。。。  

}  

@Repository 等同于配置文件中的

[html] view
plain copy

 print?

<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?

@Service(value="testService")  

public classTestService {  

   

@Resource//相当于自动装配  

private UserDao userDao ;  

   

      public UserDao getUserDao() {  

              returnuserDao;  

      }  

      public void setUserDao(UserDao userDao) {  

             this.userDao= userDao;  

      }  

   

}  

@Resource 对象间关系的组合,默认采用的是byName方式进行装配,如果根据名称查找不到关联的对象,那么会再采用byType继续查找。

 

@Service注解等同与

[html] view
plain copy

 print?

<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?

@Controller(value="ua")  

@Scope(value="prototype")  

public class UserAction {  

   

     @Resource  

     private UserService userService ;  

   

     public UserService getUserService() {  

          returnuserService;  

     }  

}  

 @Controller注解等同于

[html] view
plain copy

 print?

<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?

ServletContext application =request.getSession().getServletContext();  

ApplicationContextac = WebApplicationContextUtils.getWebApplicationContext(application);  

   

UserAction useraction = (UserAction)ac.getBean("ua");//获取控制层对象  

  

response.setContentType("text/html;charset=GBK");//设置编码  

PrintWriter out =response.getWriter();  

  

//分别将三个层的对象打印出来。  

out.println("Action:"+userAction);  

out.println("Service:"+userAction.getUserService());  

out.println("Dao:"+userAction.getUserService().getUserDao());  
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: