国内最全IT社区平台 联系我们 | 收藏本站
华晨云阿里云优惠2
您当前位置:首页 > php开源 > php教程 > Spring概念简介、bean扫描与注册实现方式

Spring概念简介、bean扫描与注册实现方式

来源:程序员人生   发布时间:2016-09-29 09:04:02 阅读次数:3069次

1、概念部份:

1、spring概念:网上有很多

2、spring核心:IOC(DI)和AOP

3、IOC:控制反转,控制权的转移,利用程序本身不负责依赖对象的创建和保护,而是由外部容器负责创建和保护,只是负责使用

解释1下就是:原来你在A类里面使用B类,需要new B(),现在不用new了,new对象的进程交给外部容器(Spring容器,它把所有的对象都称作为Bean)实现控制权转移,A类只是负责使用

4、DI:依赖注入,是IOC的1种实现方式,目的:创建对象并且组装对象之间的关系

5、创建对象并且组装对象之间的关系,这是两个进程:
1)
、创建对象可以称为bean的扫描、注册,可通过xml配置和注解两种方式实现
2)、组装对象之间的依赖关系称为注入,注入方式1般分为:setter注入和构造器注入,根据情势不同又分为xml配置注入、xml配置自动装配、注解自动装配

6、AOP:面向切面编程,具体概念略,实现看后续整理

2、bean的扫描、注册

1、xml配置(schema)方式,手动扫描

<?xml version="1.0" encoding="UTF⑻"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans⑶.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context⑶.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop⑶.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx⑶.0.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc⑶.0.xsd"> <!-- 定义注册User的bean,唯1名称为user --> <bean id="user" class="com.test.User"></bean> <!-- 定义注册Dept的bean,唯1名称为dept --> <bean name="dept" class="com.test.Dept"></bean> </beans>
2、注解方式,自动扫描

1)、现在spring的xml文件中开启注解扫描和配置扫描的范围:<context:component-scan base-package="">标签

<context:component-scan base-package="com.test"> <!-- 只扫描com.test包及子包下的注解为Service的类,而过滤注解为Controller的类 --> <context:include-filter type="annotation" expression="org.springframework.stereotype.Service"/> <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/> </context:component-scan>
延伸部份:在spring中,<context:annotation-config/>标签作用也是开启注解,它与<context:component-scan/>标签的区分是甚么(还有1个’’)???

<context:annotation-config/> 标签告知Spring到bean类中寻觅1些annotation定义的类, 比如@Autowired @PostConstruct @PreDestroy @Resource 等。
需要注意的是它并没有激活@Transactional 和 @TransactionAttribute

<context:component-scan/>标签告知Spring搜索指定包下面和1些需要被自动注入的bean,比如@Component @Repository @Service @Controller,而<context:component-scan>标签功能包括<context:annotation-config>的功能。

<mvc:annotation-driven/>这个标签的作用之1就是在springMVC中告知Spring去检测RequestMapping。其他的作用以下:
- 激活@ExceptionHandler这个annotation
- 配置了这个标签还可以将RequestMappingHandlerAdapter注册到Spring中
- 是SpringMVC提供默许的类型转化,由于我们没有在<mvc:annotation-driven/> 的属性中配置ConversionService。


1、xml配置注入,手动装配,提供setter方法或constructor构造函数

<bean id="userDao" class="com.jsun.test.springDemo.ioc.User.UserDaoImpl"></bean> <bean id="userService" class="com.jsun.test.springDemo.ioc.User.UserServiceImpl"> <!-- 配置注入属性 --> <property name="userDao" ref="userDao"></property> </bean>

//setter方式::声明UserDao属性,名字与xml中property的name名称不具有关联性,比如可以叫userDaoAlias private UserDao userDao; //setter方法,其中setXxxx的Xxxx要与xml文件中property标签中name属性值1致,首字母大写,具有关联性 public void setUserDao(UserDao userDao) { System.out.println("setUserDao注入"); this.userDao = userDao; }

<bean id="userDao" class="com.jsun.test.springDemo.ioc.User.UserDaoImpl"></bean> <bean id="userService" class="com.jsun.test.springDemo.ioc.User.UserServiceImpl"> <!-- 配置注入属性 --> <constructor-arg name="userDao" ref="userDao"></constructor-arg> </bean>

//构造器方式::声明UserDao属性,属性名与<constructor-arg>标签中name属性值无关 private UserDao userDao; //参数名与<constructor-arg>标签中name属性值具有关联性 public UserServiceImpl(UserDao userDao){ System.out.println("构造器UserServiceImpl注入"); this.userDao = userDao; }


2、xml配置注入,自动装配,提供setter方法或constructor构造函数

<bean id="userDao" class="com.jsun.test.springDemo.ioc.User.UserDaoImpl"></bean> <!-- 注册bean并自动装配所有属性bean --> <bean id="userService" class="com.jsun.test.springDemo.ioc.User.UserServiceImpl" autowire="byName/byType/constructor"></bean>

1、byName:把Bean的属性具有相同名字的的其他Bean自动装配到Bean的对应属性中,采取setter方式,set***()

2、byType:把与Bean的属性具有相同类型的的其他Bean 自动装配Bean的对应属性当中,如果存在多个该类型bean,那末抛出异常,并指出不能使用byType进行自动装配;如果没有找到相匹配的bean,则甚么事都不产生,采取setter方式

3、constructor:把与Bean的构造器入参具有相同类型的其他Bean自动装配到构造器的对应入参中,如果容器中没有找到与构造器参数类型1致的bean,那末抛出异常,采取构造器方式

注意:在利用上下文(spring容器)中,如果有多个bean的类型与该bean的自动装配属性相匹配,那末就会出错,由于byType方式只允许匹配1个类型相同的Bean。

4、补充:设置全局的默许自动装配

<beans ... default-autowire="byName"> </beans>

</pre><p></p><h2 id="3注解自动装配可以不用提供setter方法或constructor构造函数">3、注解自动装配,可以不用提供setter方法或constructor构造函数</h2><p></p><p>使用@Autowired注解,可以用在属性、setter方法、constructor构造方法上,实现自动装配。</p><p>spring的xml配置文件:</p><p></p><pre name="code" class="html"><!-- 开启注解配置 --> <context:annotation-config></context:annotation-config> <!-- 注册bean --> <bean id="userDao" class="com.jsun.test.springDemo.ioc.User.UserDaoImpl"></bean> <!-- 注册bean --> <bean id="userService" class="com.jsun.test.springDemo.ioc.User.UserServiceImpl"></bean>

1、注解在setter方法上:

2、constructor构造方法上:

3、声明的属性上:

4、补充:@Value注解基本类型数据或动态装配数据

@Value("我就是注入strValue属性的值") private String strValue;

使用表达式来动态的计算并装配属性的值,比如使用spel表达式从系统属性中获得1个值

@Value("#{systemProperties.myFavoriteSong}") private String strValue;

也能够像使用EL表达式1样,获得spring上下文中加载的*.properties资源文件数据:

@Value("${jdbc.url}") private String url;

====================================================================================================

首先了解从spring2.5增加的新特性:

这些新特性包括:注解驱动的依赖性注入(annotation-driven dependency injection),使用注解而非XML元数据来自动侦测classpath上的Spring组件,注解对生命周期方法的支持,1个新的web控制器模型将要求映照到加注解的方法上,在测试框架中支持Junit4,Spring XML命名空间的新增内容,等等。

1.条件条件   引入context 的 Schema命名空间 在配置文件中添加context:annotation-config标签

为了取得新的的特性  首先要引入新的context的Schema命名空间,该命名空间对注释驱动、属性文件引入、加载期织入等功能提供了便捷的配置。我们知道注释本身是不会做任何事情的,它仅提供元数据信息。要使元数据信息真正起作用,必须让负责处理这些元数据的处理器工作起来。

    <?xml version="1.0" encoding="UTF⑻"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx" 
xmlns:jdbc="http://www.springframework.org/schema/jdbc"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="
     http://www.springframework.org/schema/context 
     http://www.springframework.org/schema/context/spring-context⑶.2.xsd
     http://www.springframework.org/schema/beans 
     http://www.springframework.org/schema/beans/spring-beans⑶.2.xsd
     http://www.springframework.org/schema/jdbc 
     http://www.springframework.org/schema/jdbc/spring-jdbc⑶.2.xsd
     http://www.springframework.org/schema/tx 
     http://www.springframework.org/schema/tx/spring-tx⑶.2.xsd
     http://www.springframework.org/schema/aop 
     http://www.springframework.org/schema/aop/spring-aop⑶.2.xsd">

    <!--

     这个配置隐式注册了多个对注释进行解析处理的处理器,代码中就能够直接使用@Autowired, @Required等annotaion了。 

  1. AutowiredAnnotationBeanPostProcessor 对应@Autowire,CommonAnnotationBeanPostProcessor 对应@Resource等,  
  2. PersistenceAnnotationBeanPostProcessor,RequiredAnnotationBeanPostProcessor 对应@required

    -->

    <context:annotation-config />  

    </beans>

 

2.自动装配属性实例。

使用@Autowired或@Resource注解方式进行装配,这两个注解的区分是:

@Autowired 默许按类型装配,@Resource默许按名称装配,当找不到与名称匹配的bean才会按类型装配。

强烈建议 放弃@Autowire 使用@Resource 缘由 spring支持标准

Spring支持JSR⑵50注解

 

即 Java EE5中引入了“Java平台的公共注解(Common Annotations for the Java Platform)”,而且该公共注解从Java SE 61开始就被包括其中。

在2.5版本中,Spring框架的核心(core)现在支持以下JSR⑵50注解:

  • @Resource
  • @PostConstruct
  • @PreDestroy

结合Spring,这些注解在任何开发环境下都可使用——不管是不是有利用程序服务器——乃至是集成测试环境都可以。

 

使用方式:可以用来标注在字段或属性的setter方法上.如果标注在字段上,则可以省略掉该属性的getter 和setter方法。

同时

所要注入实例bean的名称可以通过@Resource的name属性指定,如果没有指定name属性,

1.当注解标注在字段上,即默许取字段的名称作为bean名称寻觅依赖对象

2.当注解标注在属性的setter方法上,即默许取属性名作为bean名称寻觅依赖对象。

 

//用于字段上  

@Resource(name="personDao")  

 private PersonDaopersonDao;

 

//用于属性的set方法上  

@Resource(name="personDao")  

public void setPersonDao(PersonDao personDao) { 

  this.personDao = personDao;  

 

注意:如果没有指定name属性,并且依照默许的名称找不到依赖对象时, @Resource注解会回退到按类型装配。但1旦指定了name属性,就只能按名称装配了。

 

3.spring自动扫描机制

spring2.5为我们引入了组件自动扫描机制,它可以在classPath路径底下寻觅标注了@Component、@Service、@Controller、@Repository注解的类,并把这些类纳入进spring容器中管理。它的作用和在xml文件中使用bean节点配置组件是1样的。

也就是要spring自动扫描机制只会查找指定类路径下包括@Component、@Service、@Controller、@Repository这4种注解的类。

要使用自动扫描机制,我们需要打开以下配置信息:

1、引入context命名空间 需要在xml配置文件中配置以下信息: 同上先引入context 命名空间,同时

2、在配置文件中添加context:component-scan标签

<context:component-scan base-package="*"/> 

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

注:

1、在使用组件扫描元素时,AutowiredAnnotationBeanPostProcessor和CommonAnnotationBeanPostProcessor会隐式地被包括进来。 也就是说,连个组件都会被自动检测并织入 - 所有这1切都不需要在XML中提供任何bean配置元数据。也就是说如果使用了context:component-scan标签,就能够不需要再引入context:annotation-config标签

 

<context:component-scan />还允许定义过滤器将基包下的某些类纳入或排除。Spring支持以下4种类型的过滤方式:

过滤器类型 表达式范例 说明 
注解 org.example.SomeAnnotation 将所有使用SomeAnnotation注解的类过滤出来 
类名指定 org.example.SomeClass 过滤指定的类 
正则表达式 com\.kedacom\.spring\.annotation\.web\..* 通过正则表达式过滤1些类 
AspectJ表达式 org.example..*Service+ 通过AspectJ表达式过滤1些类 

正则表达式为例,我罗列1个利用实例:

Java代码 
<context:component-scan base-package="com.casheen.spring.annotation"> 
    <context:exclude-filter type="regex" expression="com\.casheen\.spring\.annotation\.web\..*" />
</context:component-scan> 

        <context:component-scan base-package="com.casheen.spring.annotation">
                <context:exclude-filter type="regex" expression="com\.casheen\.spring\.annotation\.web\..*" />
        </context:component-scan>

值得注意的是<context:component-scan />配置项不但启用了对类包进行扫描以实行注释驱动Bean定义的功能,同时还启用了注释驱动自动注入的功能(即还隐式地在内部注册了AutowiredAnnotationBeanPostProcessor和CommonAnnotationBeanPostProcessor),因此当使用<context:component-scan />后,就能够将<context:annotation-config />移除。

2.为需要被扫描的类添加相应的注解,注解的类型有以下几种:

@Service用于标注业务层组件、

@Controller用于标注控制层组件(如struts中的action)、

@Repository用于标注数据访问组件,即DAO组件。

而@Component泛指组件,当组件不好归类的时候,我们可使用这个注解进行标注。 这4种注解仅仅是角色不同,但实质都1样。

细节问题总结:

1.当我们进行测试时,用ApplicationContext对象的getBean()方法寻觅组件。在之前的配置文件中我们会用<bean>标签的id属性去定义,在使用自动扫描注解后怎样取得组建的id呢?

在这类情况下,Spring会将被标注注解的类名,然后再将该类名的第1个字母变成小写,放到getBean()方法中。如:UserBizImpl类的组件Id就会是userBizImpl,获得时为context.getBean("userBizImpl");

那末,我们在使用注解时可以自定义组件的Id吗?

固然可以。我们需要在为相应的类添加注解时,在注解以后添加自定义的类名,例如:

@Service("userBiz")

public class UserBizImpl implements UserBiz {

……

}

当我们在获得该组件时,为context.getBean("userBiz);

2.在配置文件中我们可以对组件(bean)的作用域范围进行设置,它的默许值是单例模式,那末在添加注解的情况下,我们怎样设置组件的作用域范围呢?

我们可以直接在为类添加注解的同时,应用另外一个注解@Scope("prototype")来设置,以下

@Service("userBiz")@Scope("prototype")

public class UserBizImpl implements UserBiz {

……

}

3.在使用注解时,为组件设置初始化和烧毁方法:

在添加注解的相应的类中,如果想初始化或烧毁某个方法,我们可以直接在方法上添加注解,以下:

@PostConstruct

public void addItem() {

System.out.println("初始化方法");

}

@PreDestroy

public void testItem() {

System.out.println("释放资源");

}

4.在使用Spring自动扫描组件后,怎样进行依赖注入?

应用注解@Resource和@Autowired,并为依赖对象设置名称,例如:

@Resource(name="userDao")

private UserDAO userDao = null;

首先它会根据名称去找Spring自动扫描的并加入到Spring容器的组件(bean),如果有相同的名称,则进行依赖注入,如果没有相同的名称。则会根据类型区寻觅组件。

 

理解以上的内容后,你就能够很轻松的实现spirng零配置。

 

------------------------------------------------------------------------------------------------

项目后期开发工作 定义了大量的bean,现在需要为每一个数据库操作添加 日志记录,所以就定义了1个logBiz,

如果依照通常的做法,需要修改所有的配置文件 添加property属性,现在使用自动注入机制。

在baseAction中添加通用日志方法,留出1个IogBiz接口,在继承的子类action中,定义1个logBiz属性 并用@Resouce 注解。便可。





生活不易,码农辛苦
如果您觉得本网站对您的学习有所帮助,可以手机扫描二维码进行捐赠
程序员人生
------分隔线----------------------------
分享到:
------分隔线----------------------------
关闭
程序员人生