spring课堂笔记

1 什么是spring ?
    Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架.
    spring的设计思想是,单例模式和工厂模式
2 spring的四大特点(优点)

    轻量级,低侵入的设计
    Spring的DI机制降低了业务对象替换的复杂性
    spring不依赖于web容器,独立于各种应用服务器, Write Once,Run Anywhere(一次编译到处运行)
    高度开放性:Spring并不完全依赖于Spring,开发者可自由选用Spring框架的部分或全部,它的架构仍然是内在稳定的

3   spring的组成部分 七大模块

    Spring Core:Spring的基石,提供了Spring的最主要的核心机制:控制反转和依赖注入
    Spring Context:提供了Spring的上下文信息,如:国际化,验证等
   
    Spring Web支持:简化了处理多部分请求
    Spring MVC框架支持:一个全功能的构建 Web 应用程序的 MVC 实现,MVC 容纳了大量视图技术,其中包括 JSP、Velocity等。
   
    Spring Dao支持:Spring的Dao支持大大的简化了JDBC操作数据库所带来的繁琐
    Spring ORM支持 pring框架集成了若干ORM框架,从而提供了ORM的对象关系工具,其中包括 JDO、Hibernate、iBatis和TopLink。
                   所有这些都遵从Spring的通用事务和DAO异常层结构。

    Spring AOP:面向方面编程,提供了事务,日志,权限,安全等处理机制


具体解释
Spring 核心容器(Core):提供Spring框架的基本功能。核心容器的主要组件是BeanFactory,它

是工厂模式的实现。BeanFactory使用控制反转(Ioc)模式将应用程序的配置和依赖性规范与实际的应

用代码程序分开。

Spring AOP:通过配置管理特性,Spring AOP模块直接面向方面的编程功能集成到了Spring框架中,所以可以很容易的使Spring框架管理的任何对象支持 AOP。Spring AOP模块为基于Spring的应用程序

中的对象提供了事务管理服务。通过使用Spring AOP,不用依赖于EJB组件,就可以将声明性事务管理

集成到应用程序中。

Spring ORM:Spring框架集成了若干ORM框架,从而提供了ORM的对象关系工具,其中包括 JDO、Hibernate、iBatis和TopLink。所有这些都遵从Spring的通用事务和DAO异常层结构。

Spring DAO:JDBC DAO抽象层提供了有意义的异常层次的结构,可用该结构来管理异常处理和不同数据供应商抛出的异常错误信息。异常层次结构简化了错误处理,并且大大的降低 了需要编写的异常代码数量(例如,打开和关系连接)。Spring DAO的面向JDBC的异常遵从通用的DAO异常层结构。

Spring WEB:Web上下文模块建立在上下文模块(Context)的基础之上,为基于Web服务的应用

程序提供了上下文的服务。所以Spring框架支持 Jakarta Struts的集成。Web模块还简化了处理多部分请求

及将请求参数绑定到域对象的工作

Spring上下文(Context):Spring上下文是一个配置文件,向Spring框架提供上下文信息。

Spring上下文包括企业服务,例如 JNDI、EJB、电子邮件、国际化校验和调度功能。

Spring MVC:Spring的MVC框架是一个全功能的构建Web应用程序的MVC实现。通过策略接口,

MVC框架变成为高度可配置的,MVC容纳的大量视图技术,包括JSP、Velocity、Tiles、iText和Pol


4 spring的核心机制

    1).控制反转(IoC/Inverse Of Control):调用者不再创建被调用者的实例,由spring框架实现(容器创建)所以称为控制反转。
    2).依赖注入(DI/Dependence injection) :容器创建创建好实例后再注入调用者称为依赖注入。
   
    当某个角色(可能是一个Java实例,调用者)需要另一个角色(另一个Java实例,被调用者)的协助时,
    在传统的程序设计过程中,通常由调用者来创建被调用者的实例, 但在Spring里,创建被调用者的工
    作不再由调用者来完成,因此称为控制反转; 创建被调用者 实例的工作通常由Spring容器来完成,
    然后注入调用者,因此也称为依赖注入


   什么是ioc容器

    bean工厂(BeanFactory) 和 应用上下文(Application Context)


在Spring中,BeanFactory是IoC容器的核心接口。 它的职责包括:实例化、定位、配置应用程序中的对象及建立这些对象间的依赖。
                      BeanFactory
               |
               |
                  ApplicationContext(国际化,事件)
               /              /
          /                /
FileSystemXmlApplicationContext     ClassPathXmlApplicationContext

 
 常用有三种方式可以得到BeanFactory ,从而获得bean实例,并进行调用

           //        InputStream is = new FileInputStream("beans.xml");
//                InputStreamResource isr = new InputStreamResource(is);
//                BeanFactory beanFactory = new XmlBeanFactory(isr);
//               
//                Spring给出一些BeanFactory的实现类,其中最为常用的是XmlBeanFactory。
//                1、通过文件系统
//                Resource res = new FileSystemResource("src/beans.xml");
//                XmlBeanFactory beanFactory = new XmlBeanFactory(res);
//                2、通过类路径(class路径 ./表示上级目录) 
//                ClassPathResource res = new ClassPathResource("./beans.xml");
//                XmlBeanFactory beanFactory = new XmlBeanFactory(res);
//                3、通过ApplicationContext加载(ClassPathXmlApplicationContext或者FileSystemXmlApplicationContext)
//                ApplicationContext appContext = new ClassPathXmlApplicationContext(
//                new String[] {"./beans.xml"});
//                BeanFactory beanFactory = (BeanFactory) appContext;
//               
//                ClassPathResource res = new ClassPathResource("./beans.xml");
//                XmlBeanFactory beanFactory = new XmlBeanFactory(res);


         或者 ApplicationContext ctx=new FileSystemXmlApplicationContext("src/bean.xml");
       
     ApplicationContext增加功能 1国际化2事件支持
     
      国际化测试
             (1)bean配置
         <bean id="messageSource" class="org.springframework.context.support.ResourceBundleMessageSource">
                 <property name="basenames">
            <list>
                <value>message</value>
                <!-- 如果有多个资源文件,全部列在此处-->
            </list>
        </property>
          </bean>
     (2)
         ApplicationContext ctx = new FileSystemXmlApplicationContext("bean.xml");
        String[] a = {"读者"};
        String hello = ctx.getMessage("hello",a,Locale.ENGLISH);//英文
        Object[] b = {new Date()};
        String now = ctx.getMessage("hello",b,Locale.getDefault());//默认
        System.out.println(hello);
        System.out.println(now);
      
        

5  简单测试BeanFactory

   <?xml version="1.0" encoding="gb2312"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"
    "http://www.springframework.org/dtd/spring-beans.dtd">

<beans>
<bean id="aaa" class="包名.类名"/>
</beans>

   测试  FileSystemXmlApplicationContext默认从项目目录寻找bean配置文件xml
 ApplicationContext ctx = new FileSystemXmlApplicationContext("src/bean.xml");
   Test t=(Test)ctx.getBean("aaa");
 System.out.println(t.getName());
 
  依赖注入的优点,代码将更加清晰,Spring管理bean的灵巧性, bean与bean之间的依赖关系放在配置文件里组织,
  不再担心对象之间的依赖关系之后,实现更高层次的松耦合将易如反掌
 
 
  依赖注入的方式 (为什么叫依赖注入 类的四种关系  ) 
    (1)set注入(2)构造注入(3)lookup方法注入


    setter注入实例
  <bean id="aaa" class="包名.类名">
        <property name="属性名">
            <ref local="另一个bean id"/>//local与bean的区别
        </property>
    </bean>

 
 //local与bean的区别  local:当前xml   bean:整个spring容器 所走加载的xml文件搜索


  构造注入实例 提供构造方法(注入的实例参数)  要提供无参的构造方法
  <bean id="aaa" class="包名.类名">
        <constructor-arg><ref bean="steelAxe"/></constructor-arg>
    </bean>

  (3)lookup方法注入(依赖注入的bean每次都创建一个新实例)
    这种方法主要是用在Singleton的Object中使用非Singleton的Bean时,
    通过lookup-method的那个方法来取得非Singleton的Bean。一般用的不多,在用这种定义之前最佳想明白你的需求。
   
    1 Singleton的Object 中提供一个抽象方法,返回非Singleton的Bean,此时Singleton的类变成抽象类 如
      public abstract class Test//Singleton的Bean {
     TestBean tb;//非Singleton的Bean时
     
     public TestBean getTb() {
        return tb;
    }

    public void setTb(TestBean tb) {
        this.tb = tb;
    }

    public abstract TestBean  createXxx();
      }
    2  配置xml
         <bean id="testbean" class="com.TestBean"  singleton="false"/>//注意singleton="false"
        <bean id="test" class="com.Test">
         <property name="tb"> <ref local="testbean"/></property>//注意这个依然不能少
         <lookup-method name="createXxx" bean="testbean"/>
       </bean>
   3 测试
         Test t=(Test)ctx.getBean("test");
     System.out.println("不使用lookup-method注入");
     TestBean tb1=t.getTb();
     TestBean tb2=t.getTb();
     System.out.println(tb1==tb2);
     System.out.println("使用lookup-method注入");
     TestBean  tb3=t.createXxx();
     TestBean  tb4=t.createXxx();
     System.out.println(tb3==tb4);

     结果
         不使用lookup-method注入
        true
        使用lookup-method注入
        false



注入其他属性例子
    <beans>

    <bean id="aaa" class="包名.类名"/>

    <bean id="bbb" class="lee.Chinese">
        <property name="schools">
            <list>
                <value>小学</value>
                <value>中学</value>
                <value>大学</value>
            </list>
        </property>


        <property name="score">
            <map>
                <entry key="数学">
                    <value>87</value>
                </entry>
                <entry key="英语">
                    <value>89</value>
                </entry>
                <entry key="语文">
                    <value>82</value>
                </entry>
            </map>
        </property>


        <property name="health">
            <props>
                <prop key="血压">正常</prop>
                <prop key="身高">175</prop>
            </props>
        </property>


        <property name="axes">
            <set>
                <value>xxx</value>
                <bean class="包名.类名"/>
                <ref local="另外一个bean id"/>
            </set>
        </property>

    </bean>

</beans>

上述中的health指的是Properties属性类
private Properties health = new Properties();



    




  测试  Spring的DI机制降低了业务对象替换的复杂性  面向接口编程的好处  灵活配置

6  spring  bean的实例 创建的几种方式
       (1)构造器   单例思想
       (2)静态工厂(静态方法创建)

         静态方法名(iint type)

       <bean id="xxx" class="包名.类名(工厂类)" factory-method="静态方法名">
        <constructor-arg value="1"></constructor-arg>
    </bean>


    创建同一接口下不同实例

     (3)工厂方法(非静态方法创建)
           <bean id="aaa" class="包名.类名(工厂类)">
            <property name="type"><value>1</value></property>
           </bean>

       <bean id="xxx" factory-bean="工厂bean id(如刚刚的aaa) " factory-method="非静态方法名">

7  自动装配

  

       有两个bean A与B
       B作为A的一个属性,当我们没有通过setter注入或者构造注入B实例时,我们可以在A bean中指定autowire属性,
       让它在spring容器个根据不同方式自动寻找b实例


    自动装配的优缺点:

    优点:自动装配能显著减少配置的数量,自动装配可以使配置与java代码同步更新

    缺点:Spring会尽量避免在装配不明确的时候进行猜测,容器中可能存在多个bean定义跟自动装配的

    setter方法和构造器参数类型匹配


    
    byName(用于set注入) 根据属性名自动装配。此选项将检查容器并根据名字查找与属性完全一致的bean,并将其与属性自动装配。例如,在bean定义中将autowire设置为by name,而该bean包含master属性(同时提供setMaster(..)方法),Spring就会查找名为master的bean定义,并用它来装配给master属性。
 
    byType 如果容器中存在一个与指定属性类型相同的bean,那么将与该属性自动装配。如果存在多个该类型的bean,那么将会抛出异常,并指出不能使用byType方式进行自动装配。若没有找到相匹配的bean,则什么事都不发生,属性也不会被设置。如果你不希望这样,那么可以通过设置dependency-check="objects"让Spring抛出异常。
 
   constructor 与byType的方式类似,不同之处在于它应用于构造器参数。如果在容器中没有找到与构造器参数类型一致的bean,那么将会抛出异常。
 
   autodetect 通过bean类的自省机制(introspection)来决定是使用constructor还是byType方式进行自动装配。如果发现默认的构造器,那么将使用byType方式。
 
 

   在用byType时候,如果没有匹配的bean 可测试dependency-check="objects"让Spring抛出异常


 8  spring框架中bean的生命周期?

  (1)  bean定义

    在配置文件里面用<bean></bean>来进行定义。

  (2)  bean初始化

    有两种方式初始化:  也可同时使用。参数指定的后执行

         A.在配置文件中通过指定init-method属性来完成

         B.实现org.springframwork.beans.factory.InitializingBean接口  实现afterPropertiesSet()方法

  (3)  bean调用

    有三种方式可以得到bean实例,并进行调用,前面有所讲述

  (4)  bean销毁

    销毁有两种方式。也可同时使用  参数指定的销毁方法后运行

         A.使用配置文件指定的destroy-method属性

         B.实现org.springframwork.bean.factory.DisposableBean接口 实现destroy()方法
           
        可以通过调用ConfigurableListableBeanFactory(XmlBeanFactory)里的destroySingletons销毁单例对象,测试destroy()方

           ConfigurableListableBeanFactory beanFactory =
            new XmlBeanFactory(new FileSystemResource("src/bean.xml"));
             beanFactory.destroySingletons();
                }
   
     是的, 当调用beanFactory.destroySingletons()方法的时候,bean不会马上销毁,但是容器在销毁bean之前,会等到方法结束剩余的任务后
     再调用相应的销毁bean之前指定的其他方法。再干掉bean


  在 spring 2.5中 ((AbstractApplicationContext)beanFactory).registerShutdownHook();来看看bean的销毁

     原型 bean 在创建后即脱离 BeanFactory 的维护,所以只能调用初始化方法,而不能做清理工作。
 如果不是Singleton的Bean不能触发结束初始化事件和预销毁事件么?那是因为如果不是单例模式的Bean,Spring不会自己去管理,而是把它们交给Spring框架下的应用本身去管理,这样就相当于把Bean交给了GC。但是如果是单例模式的Bean,Spring会建立一个Bean的列表来统一管理,在Spring应用被关闭的时候,会执行BeanFactory中的destroySingltons()方法来逐个销毁列表中的Bean。这就是为什么只有单例模式实现Spring的那两个事件接口或指定方法属性才是有效的



      可以考虑引入bean的后处理技术



   9 spring 中bean的作用域(5种):2.0或者以前只有单例和非单例   2。0以后有以下5种

    singleton

    在每个Spring IoC容器中一个bean定义对应一个对象实例。

    prototype

     一个bean定义对应多个对象实例。

    request

    在一次HTTP请求中,一个bean定义对应一个实例;即每次HTTP请求将会有各自的bean实例, 它们依

    据某个bean定义创建而成。该作用 域仅在基于web的Spring ApplicationContext情形下有效。

    session

    在一个HTTP Session中,一个bean定义对应一个实例。该作用域仅在基于web的Spring

    ApplicationContext情形下有效。

    global session

    在一个全局的HTTP Session中,一个bean定义对应一个实例。典型情况下,仅在使用portlet

    context的时候有效。该作用域仅在基于 web的Spring ApplicationContext情形下有效。

10  spring在web环境中的配置(2种)

    1.在Web.xml中配置上下文载入器

    根据你的系统情况,你可以选择两种上下文载入器:ContextLoaderListener和ContextLoaderServlet.
    如果你的Web容器支持Servlet2.3标准或更高,你可以使用两者,否则只能使用后者.

    (1)ContextLoaderListener在Web.xml应该如下配置:
        <listener>
             <listener-class>
              org.springframework.web.con.ConttextextLoaderListener
             </listener-class>
        </listener>
    (2)ContextLoaderServlet在Web.xml应该如下配置:
        <servlet>
          <servlet-name>context</servlet-name>
          <servlet-class>
          org.springframework.web.context.ContextLoaderServlet
          </servlet-class>
          <load-on-startup>1</load-on-startup>
        </servlet>

    2.指定上下文载入器的配置文件
    不论你使用的那种上下文载入器,你都应该指明Spring配置文件的位置.如果没有指定,上下文载入器将把/web-inf/applicationContext.xml当作Spring配置文件。
    要指定Spring配置文件的位置,你可以在Servlet上下文设置contextConfigLocation参数来为上下文载入器指定一个或多个Spring配置文件
    (使用通配符或是用逗号隔开)。如下所示:
      <context-param>
        <param-name>
         contextConfigLocation
       </param-name>
       <param-value>
          /WEB-INF/bean.xml
       </param-value>
    </context-param>

    3.获得应用上下文
    接下来我们就可以获得ApplicationContext了,代码如下:
    WebApplicationContext ctx = WebApplicationContextUtils.getWebApplicationContext(servletContext);
    在一个自启动的Servlet中,我们可以这样获得它:
    public class InitialSystemServlet extends HttpServlet {
      public void init(ServletConfig config) throws ServletException {
        // 取得Spring的上下文
        WebApplicationContext ctx = WebApplicationContextUtils.getWebApplicationContext(config.getServletContext());

      }



    
   
                               测试web应用环境相关的Bean作用域

       spring 2.5 dtd配置


           <?xml version="1.0" encoding="UTF-8" ?>
       <beans xmlns="http://www.springframework.org/schema/beans"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://www.springframework.org/schema/beans
     http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">

            <bean id="test" class="com.Test" >
            </bean>
          </beans>


    如果用户使用spring的webApplicationContext,则可以使用另外3种Bean的作用域:request,session和globalSession.不过
    在使用这些作用域之前,首先必须在web容器中进行一些额外的配置,在高版本的web容器中,则可以利用HTTP请求监听器进行配置:

    <listener>
    <listener-class>
    org.springframework.web.context.request.RequestContextListener
    </listener-class>
    </listener>
   
    细心的同学可能有一个疑问:在介绍webApplicationContext初始化时,我们已经通过ContextLoaderListener将web容器与
    spring容器整合,为什么这里又要引入一个额外的RequestContextListener以支持Bean的另外3个作用域呢?
    在整合spring容器时使用ContextLoaderListener,它实现了ServletContextListener监听器接口,ServletContextListener
    只负责监听web容器启动和关闭的事件.而RequestContextListener实现ServletRequestListener监听器接口,该监听器监听
    HTTP请求事件,web服务器接收的每一次请求都会通知该监听器.
    spring容器启动和关闭操作由web容器的启动和关闭事件触发,但如果spring容器中的Bean需要request,session,globalsession
    作用域的支持,spring容器本身就必须获得web容器的HTTP请求事件,以HTTP请求的事件"驱动"Bean作用域的控制逻辑.


11 spring 对dao的支持
    <1>dao模式优点
        减少开发代码量 提高工作效率
        降低系统资源消耗 提高系统性能
        业务逻辑层与持久层(数据层)的分离,使数据与操作更为清晰。
        数据访问独立到了一层,修改具体实现类,不影响系统的架构运行
     <2>DAO模式的四个组件
        DAO接口
        DAO接口实现类
        pojo值对象
        DAO实例的定位(工厂 或 spring注入)
   

    2. 核心类 
       JdbcTemplate(需要注入DataSource)
       JdbcDaoSupport类(简化注入,有一个字段dataSource)
       并提供了getJdbcTemplate()方法获得JdbcTemplate实例
      
       DataSource接口
    为了从数据库中取得数据,我们首先需要获取一个数据库连接。Spring通过DataSource对象来完成这个工作。
    DataSource是JDBC规范的一部分,它被视为一个通用的数据库连接工厂。通过使用DataSource,
    Container或Framework可以将连接池以及事务管理的细节从应用代码中分离出来。
    作为一个开发人员,在开发和测试产品的过程中,你可能需要知道连接数据库的细节。
    但在产品实施时,你不需要知道这些细节。通常数据库管理员会帮你设置好数据源。
    在使用Spring JDBC时,你既可以通过JNDI获得数据源,也可以自行配置数据源(使用Spring提供的DataSource实现类)。
    使用后者可以更方便的脱离Web容器来进行单元测试。 这里我们将使用DriverManagerDataSource
    ,不过DataSource有多种实现, 后面我们会讲到。使用DriverManagerDataSource和你以前获取一个JDBC连接 的做法没什么两样。
    你首先必须指定JDBC驱动程序的全限定名,这样DriverManager 才能加载JDBC驱动类,接着你必须提供一个url(因JDBC驱动而异,
    为了保证设置正确请参考相关JDBC驱动的文档), 最后你必须提供一个用户连接数据库的用户名和密码。
   
    目前已经有开源的免费用了
                            DataSource
               /          /
              /            /
     DriverManagerDataSource(spring的实现)           org.apache.commons.dbcp.BasicDataSource

   
   
   
    下面我们将通过一个例子来说明如何配置一个 DataSource ,
     
     
      
      1 自己实现DataSource接口,设置数据库相关连接信息

      
      2 使用现有的实现类dataSource  编程方式 获得dataSourc 如下

    DriverManagerDataSource dataSource = new DriverManagerDataSource();
    dataSource.setDriverClassName("com.mysql.jdbc.Driver");
    dataSource.setUrl("jdbc:mysql://localhost:3306/j2ee");
    dataSource.setUsername("root");
    dataSource.setPassword("123456");

     3 spring方式

      <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property>
        <property name="url"><value>jdbc:mysql://localhost:3306/j2ee</value></property>
        <property name="username"><value>root</value></property>
        <property name="password"><value>123456</value></property>
       </bean>


      通过开源dbcp获得dataSource ,

        <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
            <property name="driverClassName">
                <value>com.mysql.jdbc.Driver</value>
            </property>
            <property name="url">
                <value>jdbc:mysql://localhost:3306/j2ee</value>
            </property>
            <property name="username">
                <value>root</value>
            </property>
            <property name="password">
                <value>123456</value>
            </property>
        </bean>




   


    使用Spring的Dao支持来写Dao层,继承JdbcDaoSupport类,JdbcDaoSupport类中有一个字段dataSource
    也就是数据库连接,因此只需继承JdbcDaoSupport类,并给它注入dataSource,就隐式的获得了数据库连接,可以在bean.xml文件中
    注入connection(即:dataSource)
     <beans>
     <!-- spring 方式获得dataSource-->
   
      <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property>
        <property name="url"><value>jdbc:mysql://localhost:3306/j2ee</value></property>
        <property name="username"><value>root</value></property>
        <property name="password"><value>123456</value></property>
       </bean>

    <bean id="xxx" class="包名.类名(dao的实现类)">
        <property name="dataSource">
            <ref local="dataSource"/>
        </property>
    </bean>

     </beans>


    3.在这个Dao 实现类中写一个内部类,使其实现:RowMapper接口,用来封装成具体的pojo对象,代码如下

     private class PersonRowMapper implements RowMapper
     {
        public Object mapRow(ResultSet resultSet, int rowNumber)
            throws SQLException {
            User user = new User();
            user.setId(resultSet.getLong("id"));
            user.setName(resultSet.getString("name"));
            user.setSex(resultSet.getString("sex"));
            user.setAge(resultSet.getInt("age"));
            user.setAddr(resultSet.getString("addr"));
            return user;
          }
     }


     4 通过JdbcTemplate类进行增删查询修改操作
         (1)增加 修改 删除示例
          public void createPerson(PersonBean p)
            {
        Object[] args = {p.getName() , new Integer(p.getAge()) };
        getJdbcTemplate().update("insert into person_test(p_name,p_age) values(?,?)", args );
            }
          (2)查询 返回单个object
            public PersonBean getPerson(int id)
            {
        Object[] args = {new Integer(id)};
        //PersonRowMapper是刚刚的内部类
        return (PersonBean)getJdbcTemplate().queryForObject("select p_name,p_age from person_test where p_id = ?", args, new PersonRowMapper());
             }
       (3)返回list
          public List findPersonsByName(String name)
          {
            //PersonRowMapper是刚刚的内部类
        return getJdbcTemplate().query("select * from person_test where p_name like '%" + name +"%'" , new PersonRowMapper());
          }


    5 spring 2.5以后提供了一个SimpleJdbcTemplate类来操作
           2.5例子
     
               public class PersonDaoSpring25 implements PersonDao {

            private SimpleJdbcTemplate simpleJdbcTemplate;

            public void setDataSource(DataSource dataSource) {
                this.simpleJdbcTemplate = new SimpleJdbcTemplate(dataSource);
            }

            @Override
            public void add(Person p) {
                // TODO Auto-generated method stub

                String sql = "insert into  person(name) values(?)";
                simpleJdbcTemplate.update(sql, p.getName());
            }

            @Override
            public void del(String id) {
                // TODO Auto-generated method stub
                String sql="delete from person where id=?";
                simpleJdbcTemplate.update(sql, id);
            }

            @Override
            public List findAll() {
                // TODO Auto-generated method stub
                String sql = "select * from person";

                List list = simpleJdbcTemplate.getJdbcOperations().query(sql,
                        new BeanPropertyRowMapper(Person.class));

                return list;

            }

            @Override
            public Person findById(String id) {
                String sql = "select * from person where id=?";

                Person p = simpleJdbcTemplate.queryForObject(sql,
                        ParameterizedBeanPropertyRowMapper.newInstance(Person.class),
                        id);

                return p;

            }

            @Override
            public void update(Person p) {
                String sql = "update person set name=? where s_id=?";

                simpleJdbcTemplate.update(sql, p.getName(), p.getId());

            }

        }


12   spring  aop    (  三种代理    五种处理   3个advisor<掌握正则>   )

      

    AOP的好处:
        消除了编码模块之间的依赖
        可以在任意阶段,向已有功能模块中填加新功能,且不侵入原有功能

    AOP的特征:

            各步骤之间的良好隔离性 , 源代码的无关性

 
  Spring 面向方面编程(AOP)   spring支持AOP功能,AspectJ更完美的支持AOP

 AOP 的核心构造是方面,它将那些影响多个类的行为封装到可重用的模块中, 将系统中通用的需求功能隔离出来形成一个功能模块,
 并将这个模块切入到应用程序当中, AOP追求的是调用者与被调用者的解耦,从而消除了OOP引起的代码混乱和分散问题,
  增强了系统的可维护性和代码的重用性
 
  方面:哪方面 系统中分为业务方面的和非业务方面  Spring主要集中于非业务方面 如(日志、事务、安全、权限等)

  1.面向方面与面向切面:AOP的实现中的几个重要的概念:
        1.1如果理解为“面向方面编程”:
            针对于某一模块进行专门的编写(如:面向日志编程,面向事务编程,面向权限等),其意思就是说面向具体的某个功能模块来编程,然后将其切入到项目中
        1.2如果理解为“面向切面编程”:
          
            连接点:程序执行过程中明确的点,如方法的调用或者异常的抛出,举个例子,如果是一个权限系统的话,要控制到类中的某一个具体的方法,那么这个方法就是一个连接点
            切入点:连接点的集合,如类中多个个方法的调用, 还是以权限系统来举例,如果要控制一个类中的多个方法的话,那么这几个方法(也即是几个连接点)组成了一个切入点
            切面:如果要控制多个类中的多个方法,那么这多个类就组成了一个切面
        连接点 --- 切入点 ---- 切面(三者之间的关系可以形象的理解为:点--线--面)
        引入(Introduction):添加方法或字段到被通知的类。Spring允许引入新的接口到任何被通知的对象。例如,
            你可以使用一个引入使任何对象实现IsModified接口,来简化缓存。
         
            AOP实现过程中的几种不同的方式:
            拦截一类中所有的方法(连接点)
            拦截一个类中的某些符合条件的方法(切入点)
            拦截多个类中的符合条件的方法(切面)


        2.1.连接点 --- 切入点 ---- 切面 --- 方面(解决方案的切入点)
        2.2.目标对象(也就是:寻找被处理的类)
        2.3.AOP代理(具体实现),分为二种,
            2.3.1  JDK动态代理(面向接口)
              1、创建一个实现java.lang.reflect.InvocationHandler 接口的代理类,如:

        import java.lang.reflect.InvocationHandler;
        import java.lang.reflect.Method;

        public class PerformanceHandler implements InvocationHandler{
            private Object target; //要进行代理的业务类的实例
            public PerformanceHandler(Object target){
            this.target = target;
            }
        //覆盖java.lang.reflect.InvocationHandler的方法invoke()进行织入(增强)的操作
            public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable{
            System.out.println("Object target proxy:"+target);
            System.out.println("模拟代理加强的方法...");
            Object obj = method.invoke(target, args); //调用目标业务类的方法
            System.out.println("模拟代理加强的方法执行完毕...");
            return obj;
            }
        }

     2、用java.lang.reflect.Proxy.newProxyInstance()方法创建动态实例来调用代理实例的方法:

        import java.lang.reflect.Proxy;

        public class Test{
            public static void main(String args[]){
            接口 xxx = new 真实实现类名(); // 在这里指定被代理类
        InvocationHandler ds = new DynamicSubject(xxx); // 初始化代理类  用代理类把目标业务类进行编织
        接口  接口变量= (接口) Proxy.newProxyInstance(xxx.getClass().getClassLoader(), xxx.getClass().getInterfaces(), ds);创建代理实例,它可以看作是要代理的目标业务类的加多了横切代码(方法)的一个子类

                       接口变量.方法//调用方法

            }
        }


                  /**
         * 从以上可以看出,动态代理可以任意指定被代理的类,即真实对象类,而代理模式则无法实现该功能,
         * 因为他把真实对象的写死在代理类里,如果要按照上述的方法使用代理模式,那么真实角色必须是事先已经存在的,并将其作为代理对象的内部属性。但是
         * 实际使用时,一个真实角色或其接口必须对应一个代理角色,如果大量使用会导致类的急剧膨胀
         */



              2.3.2 Cglib代理(面向具体类) 


             1、创建一个实现net.sf.cglib.proxy.MethodInterceptor接口的实例来为目标业务类加入进行代理时要进行的操作或增强:

            import java.lang.reflect.Method;
            import net.sf.cglib.proxy.MethodProxy;
            import net.sf.cglib.proxy.Enhancer;
            import net.sf.cglib.proxy.MethodInterceptor;
            /**
             *CGlib采用非常底层的字节码技术,可以为一个类创建子类,
             并在子类中采用方法拦截技术拦截父类方法的调用,并顺势进行增强,即是织入横切逻辑
             * @author zxl
             */
            public class CglibProxy implements MethodInterceptor{
                private Enhancer enhancer = new Enhancer();
                //覆盖MethodInterceptor接口的getProxy()方法,设置
                public Object getProxy(Class clazz){
                enhancer.setSuperclass(clazz); //设者要创建子类的类
                enhancer.setCallback(this); //设置回调的对象
                return enhancer.create(); //通过字节码技术动态创建子类实例,
                }

                public Object intercept(Object obj,Method method,Object[] args,
                    MethodProxy proxy) throws Throwable {
                System.out.println("方法执行前加入功能1");

                //通过代理类实例调用父类的方法,即是目标业务类方法的调用
                Object result = proxy.invokeSuper(obj, args);
                System.out.println("方法执行后加入功能1");
                return result;
                }
            }

            2、通过java.lang.reflect.Proxy的getProxy()动态生成目标业务类的子类,即是代理类,再由此得到代理实例:


            import java.lang.reflect.Proxy;

            public class TestCglibProxy {
                public static void main(String args[]){
                CglibProxy proxy = new CglibProxy();

                //动态生成子类的方法创建代理类
                目标类名 fsi =
                    (目标类名)proxy.getProxy(目标类名.class);

                、//调用业务方法
                }
            }










        2.4.处理(Spring AOP处理)或者叫通知,aop框架在特定的连接点执行的动作,Spring提供了五种处理方式
    2.5 Advisor=(处理+切入点)
    2.6 下面介绍Spring的处理
        实现步骤
           (1)目标对象定义 bean定义
            (2) 处理(通知)与Advisor  处理也称作通知  在使用任何pointcut切入点实现之前,你首先必须创建一个Advisor
             <1> 处理 定义(1 在哪里处理<连接点 --- 切入点 ---- 切面 --- 方面> 2什么方式处理<如前处理,后处理,前后都处理>)
                         如果只是简单的定义处理,则默认的切入点是目标类的所有方法都处理

             <2>  如果希望更精确的切入点,则建议采用Advisor(切入点+处理)
                  spring Advisor 体系
                                        Advisor
                                           |
                                           |
                                    -----------------
                                       |                 |
                                       |                 |
                                 PointcutAdvisor    IntroductionAdvistor
                                                                       |
                                       |
                                           -----------------------------------------------------------
                                                   |                   |                                     |
                           |                   |                                     |
                           DefaultPointcutAdvisor    NameMatchMethodPointcutAdvisor         RegexpMethodPointcutAdvisor




             <2.1>  以下是采用正则表达式支持类RegexpMethodPointcutAdvisor的示例
             
                  配置Advisor  (Advisor=切入点+通知)

                 <bean id="testAdvisor"
                   class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
                    <!--  advice属性确定处理bean-->
                    <property name="advice">
                        <!-- 此处的处理bean定义采用嵌套bean,也可引用容器的另一个bean-->
                        <bean class="包名.通知bean类名" />
                    </property>
                    <!--  patterns确定正则表达式模式-->
                    <property name="patterns">
                        <list>
                            <!--  确定正则表达式列表-->
                            <value>.*say.*</value> <!-- 业务实现方法名匹配 -->
                        </list>
                    </property>
                 </bean>
                      
             <2.2>  NameMatchMethodPointcutAdvisor
       
                <bean id="testAdvisor"
                   class="org.springframework.aop.support.NameMatchMethodPointcutAdvisor">
                    <!--  advice属性确定处理bean-->
                    <property name="advice">
                        <!-- 此处的处理bean定义采用嵌套bean,也可引用容器的另一个bean-->
                        <bean class="包名.通知bean类名" />
                    </property>
                    <!--  mappedName确定匹配模式-->
                    <property name="mappedName" value="hello*" />
                                      
                    <!--如果匹配多个  mappedNames确定匹配模式-->
                                        <property name="mappedNames">
                        <list>
                        <value>arrayStr1</value>
                        <value>arrayStr2</value>
                        <value>arrayStr2</value>
                        </list>
                    </property>

                 </bean>
          



             <2.2>  DefaultPointcutAdvisor示例

                            Spring2中除了NameMatchMethodPointcutAdvisor和RegexMethodPointcutAdvisor提供了对一定命名规则和正则表达式切入点的封装,
                    defaultPointcutAdvisor则提供了一切切入点的支持,Spring提供了4种用来定义切入的类,分别为:
                org.springframework.aop.support.NameMatchMethodPointcut、
                org.springframework.aop.support.JdkRegexpMethodPointcut、
                org.springframework.aop.support.Perl5RegexpMethodPointcut,
                org.springframework.aop.support.ExpressionPointcut,
               这四种可以用来描述切入点的类均可由DefaultPointcutAdvisor来进行封装,首先我们讲解一下DefaultPointcutAdvisor的使用方法:
               DefaultPointcutAdvisor类的使用很简单,他有一个 advice及 pointcut 属性,advice属性用来指明要使用的通知,pointcut属性用来指定切入点,我们可以通过构造子或设值注入方式来配置这个 Bean。
               看下面的构造子注入方式: 如下
              
                   <bean id="defaultAdvisor"       class="org.springframework.aop.support.DefaultPointcutAdvisor">
                           <property name="advice" ref=" 通知类id "/>
                           <property name="pointcut" ref="pointcutBean" />
                       </bean>
                           
                面集合这四种切入点的描述类一起讲解DefaultPointcutAdvisor:
                              <2.2.1>、org.springframework.aop.support.NameMatchMethodPointcut
   
                       <bean id="pointcutBean" class="org.springframework.aop.support.NameMatchMethodPointcut">
                        <property name="mappedNames">
                        <list>
                         <value>business*</value>
                          </list>
                       </property>
                      </bean>
                      
                               <2.2.2>  org.springframework.aop.support.JdkRegexpMethodPointcut
                    <bean id="pointcutBean"
                      class="org.springframework.aop.support. JdkRegexpMethodPointcut">
                        <property name="pattern">
                         <value>.*business.*</value>
                          </property>
                         <property name="ExcludedPattern">
                         <value>business2</value>
                          </property>
                    </bean>
                                <2.2.3> org.springframework.aop.support.Perl5RegexpMethodPointcut
                                       <bean id="pointcutBean"
                        class="org.springframework.aop.support. Perl5RegexpMethodPointcut">
                            <property name="pattern">
                             <value>.*business.*</value>
                              </property>
                             <property name="ExcludedPattern">
                             <value>business2</value>
                              </property>
                     </bean>
                                <2.2.4> org.springframework.aop.support.ExpressionPointcut

                                    <bean id="pointcutBean"
                    class="org.springframework.aop.aspectj.AspectJExpressionPointcut">
                     <property name="expression"
                        value="execution(void spring.chapter3.proxy.Component.business*(..))">
                       </property>
                   </bean>




                 <2.2.5> 自定义切入点

                        1、编写Advisor实现类

                         在此可直接使用org.springframework.aop.support.DefaultPointcutAdvisor

                        2、编写Advice实现类
                        public class PlayAdvice implements MethodBeforeAdvice{
                         public void before(Method method, Object[] args, Object target)
                           throws Throwable {
                          System.out.println("my before advice");
                         // method.invoke(target, args); 如果再调用这句,则目标方法会执行多一次
                         }
                        }

                        3、编写Pointcut实现类
                        public class PlayPointcut implements Pointcut {
                         public ClassFilter getClassFilter() {
                          return new PlayClassFilter();
                         }
                         public MethodMatcher getMethodMatcher() {
                          return new PlayMethodMatcher();
                         }
                        }
                        //PlayClassFilter的定义

                        class PlayClassFilter implements ClassFilter {
                         public boolean matches(Class clazz) {
                          if(clazz.getSimpleName().equals("Play"))
                           return true;
                          return false;
                         }
                        }

                        //PlayMethodMatcher的定义

                        class PlayMethodMatcher implements MethodMatcher {
                         public boolean isRuntime() {
                          return true;
                         }
                         public boolean matches(Method method, Class c) {
                          if(c.getSimpleName().equals("Play")&&method.getName().contains("Service"))
                           return true;
                          return false;
                         }
                         public boolean matches(Method method, Class c, Object[] args) {
                          if(c.getSimpleName().equals("Play")&&method.getName().contains("Service"))
                           return true;
                          return false;
                         }
                        }

                        4、编写目标类

                        public class Play {

                          public void playService(String what){
                          System.out.println("play "+what);
                         }
                        }


                        5、在配置文件中配置


                        <bean id="adviceBean" class="com.PlayAdvice"/>
                        <bean id="pointcutBean" class="com.PlayPointcut"/>
                        <bean id="playService" class="com.Play"/>
                        <bean
                           class="org.springframework.aop.support.DefaultPointcutAdvisor">
                           <property name="advice" ref="adviceBean"></property>
                           <property name="pointcut" ref="pointcutBean"></property>
                        </bean>
                                          
                                      
                
                    


        (3) aop代理对象定义 bean 定义    spring生成代理技术
             <0>手写aop代理(编程式)
           
                 <1>采用ProxyFactoryBean生成代理
                  特点:一种接口或者一个类 都需要对应的一个具体代理,需要指定接口或者类 以及对应的代理定义

                 <bean id="person" class="org.springframework.aop.framework.ProxyFactoryBean">
                <!--当用面向具体类实现时  proxyInterfaces属性可以不用注入 此时采用Cglib代理,必须加入Cglib的jar包 -->
                <property name="proxyInterfaces">
                    <value>包名.目标对象的接口</value>
                </property>

                <property name="target">
                    <ref local="目标对象的bean id"/>       
                </property>
                <property name="interceptorNames">
                    <list>
                        <value>处理bean id</value><!-- list说明可以注入多个处理 -->
                    </list>
                </property>
               </bean>
           
            <2> 自动代理  主要采用了 bean的后处理技术 实现了bean的后处理接口BeanPostProcessor,通常有两个常用的实现类
                      在spring bean 的生命周期中,可以在bean创建之后(如指定init-method),销毁之前(指定distory-method),让容器
              执行特定的方法,如果有更多的方法,需要在创建之时(注意是创建之时)被调用,可采用实现BeanPostProcessor接口
                        
             
             使用BeanPostProcessor,让我们的bean实现BeanPostProcessor接口
                  1     public class Chinese  implements Person,BeanPostProcessor{
                        @Override
                        public void sayHello() {
                            // TODO Auto-generated method stub
                            System.out.println("你好");
                        }

                        @Override
                            public Object postProcessAfterInitialization(Object arg0, String arg1)
                                    throws BeansException {
                                // TODO Auto-generated method stub
                            System.out.println("进入postProcessAfterInitialization方法");
                            System.out.println(arg1);//返回之前可以包装一下
                                return arg0;
                            }
                        @Override
                            public Object postProcessBeforeInitialization(Object arg0, String arg1)
                                    throws BeansException {
                                // TODO Auto-generated method stub
                            System.out.println("进入postProcessBeforeInitialization方法");
                            System.out.println(arg1);//返回之前可以包装一下
                                return arg0;
                            }
                        }
                                 
                    2 客户端调用
                                       
                        Resource res = new FileSystemResource("src/beans.xml");
                            XmlBeanFactory beanFactory = new XmlBeanFactory(res);
                        Chinese c = new Chinese();
                        beanFactory.addBeanPostProcessor(c);//注意这句哦,注册BeanPostProcessor
                            Person p = (Person)beanFactory.getBean("chinese");


                          
                        spring为我们提供了以下两种实现

                       BeanPostProcessor
                         /           /
                        /             /
              BeanNameAutoProxyCreator       DefaultAdvisorAutoProxyCreator
           
            <2.1>  采用BeanNameAutoProxyCreator自动生成代理  
                特点:多种接口或者多个类 自动创建其对应代理,需要指定接口或者类。接口或者类共用一个自动代理

                   <!-- 定义BeanNameAutoProxyCreator-->
                 <bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
                <!--  指定对满足哪些bean name的bean自动生成业务代理 -->
                    <property name="beanNames">
                    <!--  下面是所有需要自动创建代理的bean-->
                    <list>
                    <value>bean id 1</value>
                    <value>bean id 2</value>
                    </list>
                    </property>
                <!--  下面定义BeanNameAutoProxyCreator所需的处理-->
                <property name="interceptorNames">
                    <list>
                    <value>处理 bean 1</value>
                    <value>处理 bean 2</value>
                    </list>
                 </property>
                </bean>
                   
           
            <2.2> 更牛的自动代理DefaultAdvisorAutoProxyCreator, 面向接口
                 特点:多种接口或者多个类 自动创建其对应代理,不需要指定接口或者类,提供Advisor(切入点+通知),默认搜索所有的Advisor 如前面的正则表达式
                <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator"></bean>


        (1).before advice(前置通知)   implements MethodBeforeAdvice  重写before()方法
           是在目标bean方法调用前调用,没有返回值,通常意外情况下,会继续运行下一步方法.记住的一点是没有返回 
             例子如下
         public class MyBeforeAdvisor implements MethodBeforeAdvice
        {
         
            public void before(Method m, Object[] args, Object target) throws Throwable
            {
                System.out.println("方法调用之前...");
                System.out.println("下面是方法调用的信息:");
                System.out.println("所执行的方法是:" + m);
                System.out.println("调用方法的参数是:" + args);
                System.out.println("目标对象是:" + target);
                System.out.println("方法调用之前111...");
            }
        }
       

    (2).after advice(后置通知)   implements  AfterReturningAdvice 重写afterReturning()方法
    是在方法调用后调用,有返回值,记住的一点是有返回值。
        public void afterReturning(Object returnValue, Method m, Object[] args, Object target)throws Throwable
       {
        System.out.println("方法调用结束...");
        System.out.println("目标方法的返回值是 : " + returnValue);
        System.out.println("目标方法是 : " + m);
        System.out.println("目标方法的参数是 : " + args);
        System.out.println("目标对象是 : " + target);

           }
    (3).around advice(环绕通知)  implements MethodInterceptor 重写其内的invoke()方法,实现方式与Struts2的拦截器一模一样
         调用其 MethodInvocation 的proceed()方法;
            例子如下
          public class MyAroundInterceptor implements MethodInterceptor
        {
         
            public Object invoke(MethodInvocation invocation) throws Throwable
            {
           
            System.out.println("调用生小孩方法之前娶老婆: invocation对象:[" + invocation + "]");
            Object rval = invocation.proceed();
            System.out.println("调用方法之后...坐月子");
            return rval;
            }
        }
 

    (4).throws advice(异常通知)  implements ThrowsAdvice
       例子如下
         public class MyExceptionAdvice implements ThrowsAdvice
        {
            public void afterThrowing(ClassNotFoundException ex) throws Throwable
            {
            System.out.println("系统抛出ClassNotFoundException异常,异常提示为: " + ex.getMessage());
            }
         
            public void afterThrowing(Method m,Object[] o,Object target,Throwable e)throws Throwable{
               System.out.println("消息:"+e.getMessage());
            }

            public void afterThrowing(Method m, Object[] args, Object target, ArithmeticException ex)
            {
            System.out.println("系统抛出ArithmeticException异常,异常提示为: " + ex.getMessage());
            System.out.println("抛出异常的方法为: " + m);
            System.out.println("抛出异常的方法的参数为: " + args);
            System.out.println("抛出异常的目标对象为: " + target);
            }

         
        }


    (5).introduce advice(引入通知)  理解引入通知是干什么的
    引入通知是一种特殊的通知,它能将新的成员变量、成员方法引入到目标类中,引入新的接口。它不能作用于任何切入点,
    因为它只作用于类层次,而不是方法层次。实现引入通知需要实现IntroductionAdvisor和IntroductionInterceptor接口。
    引入通知不能调用proceed方法。Advisor必须针对每个实例,并且是有状态的。
    引入通知的效果类似于设计模式中的访问者模式(Visitor Pattern)
   
    示例:为目标对象加锁 即当目标对象企图修改某个属性值时,无法修改
        <1>extends DefaultIntroductionAdvisor,因为DefaultIntroductionAdvisor,因为实现IntroductionAdvisor实现IntroductionAdvisor
                   
             public class LockMixinAdvisor extends DefaultIntroductionAdvisor
            {
                public LockMixinAdvisor()
                {
                super(new LockMixin(), Lockable.class);//Lockable是我们要引入的接口。LockMixin是它的实现类
                }
            }
             <2>extends DelegatingIntroductionInterceptor,因为DelegatingIntroductionInterceptor实现了IntroductionInterceptor接口
                   public class LockMixin extends DelegatingIntroductionInterceptor implements Lockable
            {
                private boolean locked;
                public void lock()
                {
                this.locked = true;
                }
                public void unlock()
                {
                this.locked = false;
                }
                public boolean locked()
                {
                return this.locked;
                }
                public Object invoke(MethodInvocation invocation) throws Throwable
                {
                if (locked() && invocation.getMethod().getName().indexOf("set") == 0)
                    throw new Exception("属性加锁,无法修改");
                return super.invoke(invocation);
                }
            }
            
        //   Lockable接口
            public interface Lockable
                {
                    void lock();
                    void unlock();
                    boolean locked();
                }

                《3》客户端调用 示例
                    ApplicationContext ctx = new FileSystemXmlApplicationContext("bean.xml");
                        Person p = (Person)ctx.getBean("person");
                        p.run();
                        p.setAge(56);
                        p.run();
                    Lockable lp = (Lockable)p;//注意这里的强转,表明引入成功
                        lp.lock();
                        p.setAge(23);
                    p.run();



     3.spring 对aspectj的支持  基于Schema的AOP支持      更细粒度的控制      需要aspectjweaver.jar支持
         相应dtd
             
         <?xml version="1.0" encoding="UTF-8"?>
            <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"
            xmlns:aop="http://www.springframework.org/schema/aop"
            xsi:schemaLocation="http://www.springframework.org/schema/beans  
               http://www.springframework.org/schema/beans/spring-beans-2.5.xsd  
               http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd  
               http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd">
            <aop:aspectj-autoproxy />  <!-- 启动aspectj支持-->
              
               <aop:config>
                <aop:aspect id="asp" ref="拦截的处理bean id <如包含前置通知,后置通知,环绕通知等>">
                    <aop:pointcut id="mycut"
                        expression="execution(* com.*.*(..))" />
                    <!-- pointcut-ref :切入点引用-->
               
                        <aop:before pointcut-ref="mycut" method="before处理方法名" />
                    <aop:after-returning pointcut-ref="mycut"
                        method="后置处理方法名" returning="返回参数名" />
                    <aop:after-throwing pointcut-ref="mycut" method="异常处理方法名"
                        throwing="异常处理方法参数名" />
                    <aop:after pointcut-ref="mycut" method="最终通知方法名" />
                    <aop:around pointcut-ref="mycut" method="环绕通知方法名" />

                   </aop:aspect>
                    </aop:config>
              
             </beans>   


          切面代码实例
          public class MyInterceptor {

        private void anyMethod() {
        }// 声明一个切入点

        public void doAccessCheck() {
            System.out.println("前置通知:" );
        }

        public void doAfterReturning(int result) {
            System.out.println("后置通知:" );
        }

        public void doAfterThrowing(Exception e) {
            System.out.println("例外通知:" + e);
        }
       
        public void doAfter() {
            System.out.println("最终通知");
        }

        public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable {
            // if(){//判断用户是否在权限
            System.out.println("进入方法");
            Object result = pjp.proceed();// 当使用环绕通知时,这个方法必须调用,否则拦截到的方法就不会再执行了
            System.out.println("退出方法");
            // }
            return result;
        }

          }


    Spring AOP 用户可能会经常使用 execution切入点指示符。执行表达式的格式如下:

     execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern)
          throws-pattern?)

     execution(修饰符? 返回类型  包名。类名?  方法名(参数)异常抛出?)

       下面给出一些通用切入点表达式的例子。

        任意公共方法的执行:
        execution(public * *(..))

        任何一个名字以“set”开始的方法的执行:
        execution(* set*(..))

        AccountService 接口定义的任意方法的执行:
        execution(* com.xyz.service.AccountService.*(..))

        在service包中定义的任意方法的执行:
        execution(* com.xyz.service.*.*(..))

        在service包或其子包中定义的任意方法的执行:
        execution(* com.xyz.service..*.*(..))

        如果只拦截返回方法为String类型的,表达式应为::
        execution(java.lang.String com.xyz.service..*.*(..))

        如果只拦截第一个输入参数为String类型的方法,表达式应为::
        execution(* com.xyz.service..*.*(java.lang.String,..))

        如果只拦截非void返回类型的方法,表达式应为::
        execution(!void  com.xyz.service..*.*(..))




  

     4 利用aop进行事务编程  Spring事务配置的五种方式 (掌握Spring事务体系图)
  
   传统上,J2EE开发者有两个事务管理的选择: 全局 或 本地事务(局部事务)。全局事务由应用服务器管理,使用JTA。
   局部事务是和资源相关的,比如一个和JDBC连接关联的事务
   
      事务种类分为: jdbc事务(局部事务) jta事务(全局事务) 容器事务

    1、JDBC事务控制的局限性在一个数据库连接内,但是其使用简单。局部事务

    2、JTA事务的功能强大,事务可以跨越多个数据库或多个DAO,使用也比较复杂。全局事务

    3、容器事务,主要指的是J2EE应用服务器提供的事务管理,局限于EJB应用

   两种事务的比较
             1、JDBC事务控制的局限性在一个数据库连接内,但是其使用简单。

             2、JTA事务的功能强大,事务可以跨越多个数据库或多个DAO,使用也比较复杂 

 



   从实现的角度划分

       Spring把事务分为两种一种是:编程式事务,一种是声明式事务。
       编程式事务是侵入式事务 比较灵活,编程式事务则要操作逻辑代码。存在重复的代码比较多,相对繁琐,而且不利于系统的扩展;
       声明式事务是非侵入式的事务。声明式事务只需在配置文件中配置,而不需要去操作逻辑代码。

  spring事务的7大传播属性

    1.PROPAGATION_MANDATORY:要求调用该方法的线程必须处于事务环境中,否则抛出异常。
    2.PROPAGATION_NESTED:如果执行该方法的线程已处于事务环境下,依然启动新的事务,方法在嵌套的事务里执行。
      如果执行该方法的线程并未处于事务中,也启动新的事务,然后执行该方法,此时与PROPAGATION_REQUIRED相同。
    3.PROPAGATION_NEVER:不允许调用该方法的线程处于事务环境下,如果调用该方法的线程处于事务环境下,则抛出异常。
    4.PEOPAGATION_NOT_SUPPORTED:如果调用该方法的线程处于在事务中,则先暂停当前事务,然后执行该方法。
    5.PEOPAGATION_REQUIRED:要求在事务环境中执行该方法,如果当前执行线程已处于事务中,则直接调用,
    如果当前执行线程不已处于事务中,则启动新的事务后执行该方法。
    6.PEOPAGATION_REQUIRES_NEW:该方法要求有一个在新的事务环境中执行,如果当前执行线程已处于事务中,
    先暂停当前事务,启动新的事务后执行该方法;如果当前执行线程不已处于事务中,则启动新的事务后执行该方法。
    7.PEOPAGATION_SUPPORTS:如果当前执行线程已处于事务中,则使用当前事务,否则不使用事务。



         Spring配置文件中关于事务配置总是由三个组成部分,分别是DataSource、TransactionManager和代理机制这三部分,无论哪种配置方式,一般变化的只是代理机制这部 DataSource、TransactionManager这两部分只是会根据数据访问方式有所变化,比如使用Hibernate进行数据访问时,DataSource实际为SessionFactory,TransactionManager的实现为HibernateTransactionManager。根据代理机制的不同,总结了五种Spring事务的配置方式
          现在以dataSource为例,当然要先配置dataSource,参见去前面文档spring方式获得dataSource部分
      <1> 每个bean有一个代理
         
           <1.1>配置事务管理器transactionManager
        
            <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
                   <!-- 给DataSourceTransactionManager注入dataSource-->
          <property name="dataSource"><ref local="dataSource"/></property>
                 </bean>
               <1.2>配置事务代理bean    TransactionProxyFactoryBean
         <bean id="xxx" 
        class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"> 
           <!-- 配置事务管理器  引用上面配置的来注入 --> 
           <property name="transactionManager" ref="transactionManager" />    
           <property name="target" ref="操作数据库的目标 dao bean id" /> 
        <!-- 配置事务属性 --> 
        <property name="transactionAttributes"> 
            <props> 
            <prop key="*">PROPAGATION_REQUIRED</prop>
            </props> 
        </property> 
        </bean> 

         <2>  所有Bean共享一个代理基类   实则上就是spring 的模板机制  配置一个模板,使其为abstract="true"
                 spring中bean 的继承 在java 类中称为继承 在spring称为模板
          

         <2.1>代理定义 定义模板(父类)
             
          <bean id="baseTxProxy" lazy-init="true" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean" abstract="true">
        <property name="transactionManager"><ref bean="transactionManager"/></property>
        <property name="transactionAttributes">
            <props>
            <prop key="*">PROPAGATION_REQUIRED</prop>
            </props>
          </property>
         </bean>

          <2.2> 为不同的dao操作类加上事务
            <!-- 给dao1加事务-->
          <bean id="daotran1" parent="baseTxProxy">     <!-- baseTxProxy为刚刚配置的模板-->
            <property name="target">
                <ref local="dao1"/>
            </property>
            </bean>
         
            <!-- 给dao2加事务-->
          <bean id="daotran2" parent="baseTxProxy">
            <property name="target">
                <ref local="dao2"/>
            </property>
            </bean>
           
           
           
            <!--  原来的两步骤-->
            <bean id="personDaoSpring25" class="com.PersonDaoImplSpring25">
            <property name="dataSource"><ref local="springdataSource"/></property>
            </bean>


            <bean id="personDaoSpring25Tran" parent="parent">
            <property name="target"><ref local="personDaoSpring25"/></property>
            </bean>

 
            <!--  原来的两步骤 变作一步-->

            <bean id="personDaoSpring25One"  parent="parent">
              <property name="target">
              <bean class="com.PersonDaoImplSpring25">
               <property name="dataSource"> <ref local="springdataSource"/></property>
              </bean>
              </property>
            </bean>
           
           
 
           <3>使用拦截器
          <3.1>定义事务拦截器  org.springframework.transaction.interceptor.TransactionInterceptor
             <bean id="transactionInterceptor" 
            class="org.springframework.transaction.interceptor.TransactionInterceptor"> 
            <!-- 配置事务管理器  引用上面配置的来注入 --> 
            <property name="transactionManager" ref="transactionManager" /> 
            <!-- 配置事务属性 --> 
            <property name="transactionAttributes"> 
                <props> 
                <prop key="*">PROPAGATION_REQUIRED</prop> 
                </props> 
            </property> 
            </bean>
               <3.2> 定义代理 bean 采用自动代理 BeanNameAutoProxyCreator (前面讲过有例子)
                     <bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"> 
            <property name="beanNames"> 
                <list> 
                <value>*dao</value> <!--  dao bean id匹配  -->
                </list> 
            </property> 
            <property name="interceptorNames"> 
                <list> 
                <value>transactionInterceptor</value>  <!--  引用事务拦截器  -->
                </list> 
            </property> 
                   </bean> 


         <4> 使用tx标签配置的拦截器  基于Schema的AOP支持
             <?xml version="1.0" encoding="UTF-8"?>
        <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"
        xmlns:aop="http://www.springframework.org/schema/aop"
        xmlns:tx="http://www.springframework.org/schema/tx"
        xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
           http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-2.5.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
           http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">




           <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="*" propagation="REQUIRED" />
        </tx:attributes>
               </tx:advice>

         <aop:config>

            <aop:pointcut   id="interceptorPointCuts"
                expression="execution(* com.*.*(..))"   />
            <aop:advisor advice-ref="txAdvice"
                pointcut-ref="interceptorPointCuts" />    
                 
        </aop:config>  

          <5> 使用注解
           配置文件加上
            
         <context:annotation-config />  <!--  启用注解支持  -->
                 <context:component-scan base-package="com" /><!--  要扫描的com包  -->
         <tx:annotation-driven transaction-manager="transactionManager"/> <!--  使用tx标签引入事务管理器  -->
             
           需要事务的类加上@Transactional注解
              



13  spring与struts的整合   


     整合中要思考的是:怎样才能加载和实例化spring容器.先把spring容器放到web容器中,然后再根据一些方法去提取,
    
    
     13.1  在Strust1中直接使用spring
    
     让我们回顾一下前面所讲的如何在web环境中使用spring,
     根据servlet版本的不用,我们可以通过在web.xml里配置servlet或者监听器来实例化spring容器,
     
        1.在web.xml中配置
    ①.加载spring 容器
    <!-- 加载spring 容器 -->
    <context-param>
      <param-name>contextConfigLocation</param-name>
      <param-value>/WEB-INF/classes/beans.xml</param-value>
    </context-param>
    ②.实例化spring 容器
    <!-- 初始化spring容器方法一  servlet版本 前面有讲述-->
    <servlet>
    <servlet-name>contextLoader</servlet-name>
    <servlet-class>org.springframework.web.context.ContextLoaderServlet</servlet-class>
    <load-on-startup>0</load-on-startup>
    </servlet>
    <!-- 初始化spring容器方法二   监听器版 前面有讲述-->
    <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>
    
    
     之后就可以通过WebApplicationContextUtils取得Spring的上下文

        WebApplicationContext ctx = WebApplicationContextUtils.getWebApplicationContext(当前ServletContext);
      
        如在struts1中,必须知道当前的servlet上下文
             ①WebApplicationContext ctx=WebApplicationContextUtils.getWebApplicationContext(this.servlet.getServletContext());
         //这种获得servlet上下文在以下整合的后两种方式中会呗空指针错误,所以采用request.getSession().getServletContext()获得servlet上下文。如
         ②WebApplicationContext beanFactory=WebApplicationContextUtils.getWebApplicationContext(request.getSession().getServletContext());
          
       如在struts2中
        WebApplicationContext ctx=WebApplicationContextUtils.getWebApplicationContext();
 

        有了WebApplicationContext实例,我们就可以通过getBean方法获得spring容器中的bean了,如下所示
                    ctx.getBean("id名")
      

     
      13.2  Strust1整合Spring的其他3种方式
              
             在上面的Strust1中直接使用Spring,我们是通过在web.xml中配置servlet或者是listener加载和实例化spring容器。实则上
          spring提供了一种专门针对于struts1的插件来加载和实例化spring容器,只需在struts-config.xml加入即可,
          其他3种整合方式中可采用以下插件配置替代在web环境中的监听器或者servlet
                    
            <plug-in className="org.springframework.web.struts.ContextLoaderPlugIn">
                <set-property property="contextConfigLocation"
                    value="/WEB-INF/classes/applicationContext.xml" />
            </plug-in>

                          <!--
              注意 /WEB-INF/classes/applicationContext.xml这个位置,如果你的文件在WEB-INF。则写成/WEB-INF/applicationContext.xml
              在eclipse 中 /WEB-INF/classes/applicationContext.xml对应的目录是src目录,applicationContext.xml放在src下
              -->

                  ◆使用Spring的ActionSupport类整合Structs
              优点:
            1)  简单
              缺点:
            1)  耦合高  它将Struts动作与Spring框架耦合在一起。如果您想替换掉Spring,那么您必须重写代码
            2)  违反IOC
            3)  无法使用 DispatchAction
          ◆使用Spring的DelegatingRequestProcessor覆盖Struts1的RequestProcessor
             DelegatingRequestProcessor方法的确比第一种方法好,缺点 RequestProcessor如果改变了怎么办或者是有不同的RequestProcessor怎么办呢
          ◆将Struts Action管理委托给Spring框架
         动作委托解决方法是这三种方法中最好的。Struts动作不了解Spring,不对代码作任何改变就可用于非Spring应用程序中。RequestProcessor的改变不会影响它,并且它可以利用Spring AOP特性的优点。
                     优点
            1.  动作委托解决方法是这三种方法中最好的。
            2.  不使用Spring api编写  Action
            3.  利用了IOC装配


            三种方式的相关配置:
          
            <1> Action直接继承ActionSupport,在Action类的execute()方法中通过
            ApplicationContext ctx=getWebApplicationContext()获得Spring的
            上下文,通过上下文获取相应的bean实例
              
            <2>覆盖RequestProcessor
            1.在struts-config.xml中配置控制器

         <controller processorClass="org.springframework.web.struts.DelegatingRequestProcessor"/>
            2.在applicationContext.xml文件中注册action动作bean

            <bean name="/struts-config.xml配置的对应action的path" class="Action类的包名类名" >
            <!--注意上面的是name而不是大家经常写的id-->
                <property name="service">
                    <ref bean="service" />  <!--注入spring实例,假设属性名为service-->
                </property>
            </bean>
            3.Action类中声明一个Service私有属性及其setter方法,通过IOC注入Service实例。
              
            <3>动作管理委托给Spring
            1.struts-config.xml中action标签内的type属性值替换为org.springframework.web.struts.DelegatingActionProxy,如下
             
                 <action input="/index.jsp" path="/hello" scope="session"
                   type="org.springframework.web.struts.DelegatingActionProxy">
                <forward name="success" path="/success.jsp" />
                 </action>

            2.在applicationContext.xml文件中注册action动作bean

            <bean name="/struts-config.xml配置的对应action的path" class="Action类的包名.类名" >
                <property name="service">
                    <ref bean="service" />
                </property>
            </bean>
            3.Action类中声明一个Service私有属性及其setter方法,通过IOC注入Service实例



  13.2  Spring与Struts2的整合
       由于Struts2得强大可插拔式设计,使得整合spring的工作变得异常简单,struts2-spring-plugin-2.0.8.jar就是支持spring的插件包
        

            0.把struts2支持spring的插件包jar拷贝到项目中
         
        1.web.xml的配置:

 <?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">

    <!-- 监听器 -->
    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>

    <!-- 定义Struts2的FilterDispathcer的Filter -->
    <filter>
        <filter-name>struts2</filter-name>
        <filter-class>org.apache.struts2.dispatcher.FilterDispatcher</filter-class>
    </filter>

    <!-- FilterDispatcher用来初始化struts2并且处理所有的WEB请求。 -->
    <filter-mapping>
        <filter-name>struts2</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
</web-app>

    2.applicationContext.xml的配置(WEB-INF目录下)
        (如果不使用默认的 则可以在web.xml通过以下配置修改xml的配置以及名称)
      <!-- 加载spring 容器 -->
    <context-param>
      <param-name>contextConfigLocation</param-name>
      <param-value>/WEB-INF/classes/beans.xml</param-value>
    </context-param>
    
     



 <?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">

    <bean name="service" class="com.service.impl.ServiceImpl"/>

    <!-- action的配置 -->
    <bean name="abc" class="com.LoginAction" >
        <property name="service">
            <ref bean="service" /> <!--注入spring实例,假设action类中有一个属性名为service-->
        </property>
    </bean>
</beans>

    3.struts.xml的配置

 <?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
        "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
        "http://struts.apache.org/dtds/struts-2.0.dtd">
<struts>
    <include file="struts-default.xml" />
    <package name="default" extends="struts-default">
        <!-- action标签中的class属性与applicationContext.xml中的action的bean的name属性一致 如以下的abc -->
        <action name="login" class="abc">
            <result name="input" >/login.jsp</result>
            <result name="success">/welcome.jsp</result>
            <result name="error">/failed.jsp</result>
        </action>
    </package>
</struts>

    4.action类中声明service的一个变量及setter方法


           
14  Spring2.5有哪些改进


    Spring2.5rc1发布了,一直想知道它葫芦里卖什么药
    0 spring对dao的支持2.5的simpleJdbcTemplate与2.0的JdbcTemplate
    1 spring   bean的作用域 已经有原来的单例和非单例 变成了5个
        2 Jar 包:一个包分解成多个包 如 SpringMVC 不再使用spring.jar 这个庞大的包了。 Spring-webmvc.jar, spring-webmvc-portlet.jar 可以在 lib/modules 目录下找到,而且对于Struts1.* 的支持的包也应该改成 spring-webmvc-struts.jar了。
    3   Hibernate:注意Spring已经不再支持Hibernate 3.1之前的版本了,也就是说2.1,3.0的版本也无法使用了。如果仍然要用的话,推荐使用Spring的2.0.6/2.0.7版本。
    4. JDK版本: JDK至少要1.4.2以上,如果仍然打算使用1.3的话就请用2.0.6/2.0.7版本吧。
    5. XML配置:推荐使用XML metadata格式,不过先前的DTD仍然支持。需要注意的一点是,Spring2.0的DTD继续使用“singleton”属性,但是新的Spring2.5 DTD不允许使用“singleton”属性了,改成“scope”属性来描述bean的生命周期。
    6. 废弃的类与方法:有些在先前版本标注过“@deprecated”的类与方法已经完全被废弃了。
    比如:

       1.
          ResultReader : 被 RowMapper 接口取代。
       2.
          BeanFactoryBootstrap : 考虑使用 BeanFactoryLocator 或自定义的bootstrap类来代替。

    7. Apache OJB:注意Spring源代码中已经完全把这个去掉了,但与之整合的类仍然可以在Spring Modules project找到。https://springmodules.dev.java.net/
    8. iBATIS:注意Spring已经不再支持iBATIS 的1.3版本了, 如果想继续的话,请升级到2.3以上吧。
    9. JDO:注意Spring已经不支持JDO1.0了。同样,你可以升级到2.0以上版本,或者使用Spring的2.0.6/2.0.7版本。
    10. UrlFilenameViewController:这个东东我也没有接触过,这里,我就简单翻译一下吧。
    “考虑到嵌套路径的请求,视图名称现在决定叫UrlFilenameViewController了,这是一个突破性的改变,并且意味着如果你从Spring1.* 升级到Spring2.0 你可以使用该类从而减少Spring Web MVC的配置工作。”

15 spring 注解

     <一> 1  @Autowired 按照类型进行装配  2   @Autowired              
                                              @Qualifier("bean名称")
         
        需要<bean class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor"/>  
     <二> JSR-250 的注释  
         @Resource  不用于构造注入         有2个属性name与type   默认 byName 自动注入
          
           使用示例   @Resource(type=Chinese.class)  @Resource(name="chinese")


         @PostConstruct
         @PreDestroy
           加入common-annotations.jar包
           需要<bean  class="org.springframework.context.annotation.CommonAnnotationBeanPostProcessor"/>


     可用 <context:annotation-config/>。简化第一和第二的配置 请看下面的配置:

            <?xml version="1.0" encoding="UTF-8" ?>
            <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-2.5.xsd
             http://www.springframework.org/schema/context
             http://www.springframework.org/schema/context/spring-context-2.5.xsd">
             <context:annotation-config/>

            <bean id="chinese" class="com.Chinese"></bean>
            <bean id="American" class="com.American"></bean>
            <bean id="test" class="com.Test"></bean>

            </beans>

     <三>  @Component
        
      Component或者@Component("bean名")    @Component默认bean名为类名  首字母小写
          不需要配置bean  配置文件如下
                           <?xml version="1.0" encoding="UTF-8" ?>
        <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-2.5.xsd
         http://www.springframework.org/schema/context
         http://www.springframework.org/schema/context/spring-context-2.5.xsd">
            <context:component-scan base-package="com"/>   <!--com包  这里改成你的包名-->
        </beans>
         

    <四> 采用具有特殊语义的注释

    Spring 2.5 中除了提供 @Component 注释外,还定义了几个拥有特殊语义的注释,它们分别是:@Repository、@Service 和 @Controller。
    在目前的 Spring 版本中,这 3 个注释和 @Component 是等效的,但是从注释类的命名上,很容易看出这 3 个注释分别和持久层、
    业务层和控制层(Web 层)相对应。
    虽然目前这 3 个注释和 @Component 相比没有什么新意,但 Spring 将在以后的版本中为它们添加特殊的功能。
    所以,如果 Web 应用程序采用了经典的三层分层结构的话,最好在持久层、业务层和控制层分别采用 @Repository、@Service 和 @Controller
    对分层中的类进行注释,而用 @Component 对那些比较中立的类进行注释。

    <五> @AspectJ注解  @Aspect  @Pointcut  @Before    
          
        1 @Pointcut  @Before 分开

        //定义切入点
         @Aspect
          public class SampleAspect {
              @Pointcut("execution(* com..*.*(..))")
              public void inServiceLayer() {
              System.out.println("33333333333");
              }
          }
    

            //处理类(通知类)
         @Aspect
        public class SampleAdvice {
            @Before("com.SampleAspect.inServiceLayer()")
            public void logInfo() {
                System.out.println("=====================================");
                System.out.println("Aop: do before in service layer");
                System.out.println("=====================================");
            }
        }


      2 @Pointcut  @Before 结合用
         
         //①通过该注解将PreGreetingAspect标识为一个切面
         @Aspect
         public class PreGreetingAspect{
        // ②定义切点和增强类型
         @Before("execution(* greetTo(..))")
         public void beforeGreeting()
            {
                //③增强的横切逻辑
                System.out.println("How are you");
            }
        }


       

                    以下是具体说明  高手可掠过
      



    Spring 通过一个 BeanPostProcessor 对 注解进行解析,所以要让 注解 起作用必须事先在 Spring 容器中声明 AutowiredAnnotationBeanPostProcessor Bean。
 
  
   如下
     <!-- 该 BeanPostProcessor 将自动起作用,对标注 @Autowired 的 Bean 进行自动注入 -->

   <bean class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor"/>

   15.1    Spring 2.5 引入了 @Autowired 注解,它可以对类成员变量、方法及构造函数进行标注,完成自动装配的工作,按照类型进行装配
          如
             @Autowired
        public void setP(Person p) {
            this.p = p;
        }


    15.2   当不存在不匹配的bean或者是匹配的bean的数目大于1时,按照类型进行装配都会报出异常,可采用以下方式不报都会报出异常
               @Autowired(required = false)  


    15.3    采用以下方式指定bean,按名字匹配
             @Autowired
             @Qualifier("bean名称")

         用在构造方法上
           @Autowired
        public Test(@Qualifier("chinese")Person p) {
            // TODO Auto-generated constructor stub
            this.p=p;
        }




    15.3 使用 JSR-250 的注释   @Resource   @PostConstruct 和 @PreDestroy


            Spring 不但支持自己定义的 @Autowired 的注释,还支持几个由 JSR-250 规范定义的注释,它们分别是 @Resource、@PostConstruct 以及 @PreDestroy。
           
        为了让 JSR-250 的注释生效,除了在 Bean 类中标注这些注释外,还需要在 Spring 容器中注册一个负责处理这些注释的 BeanPostProcessor:

            <bean  class="org.springframework.context.annotation.CommonAnnotationBeanPostProcessor"/>


        @Resource

    @Resource 的作用相当于 @Autowired,只不过 @Autowired 按 byType 自动注入,而 @Resource 默认按 byName 自动注入罢了。@Resource 有两个属性是比较重要的,分别是 name 和 type,Spring 将 @Resource 注释的 name 属性解析为 Bean 的名字,而 type 属性则解析为 Bean 的类型。所以如果使用 name 属性,则使用 byName 的自动注入策略,而使用 type 属性时则使用 byType 自动注入策略。如果既不指定 name 也不指定 type 属性,这时将通过反射机制使用 byName 自动注入策略。
    Resource 注释类位于 Spring 发布包的 lib/j2ee/common-annotations.jar 类包中,因此在使用之前必须将其加入到项目的类库中。

           
        JSR-250 为初始化之后/销毁之前方法的指定定义了两个注释类,分别是 @PostConstruct 和 @PreDestroy,这两个注释只能应用于方法上。标注了 @PostConstruct 注释的方法将在类实例化后调用,而标注了 @PreDestroy 的方法将在类销毁之前调用。


        使用 <context:annotation-config/> 简化配置
       而我们前面所介绍的 AutowiredAnnotationBeanPostProcessor 和 CommonAnnotationBeanPostProcessor 就是处理这些注释元数据的处理器。但是直接在 Spring 配置文件中定义这些 Bean 显得比较笨拙。Spring 为我们提供了一种方便的注册这些 BeanPostProcessor 的方式,这就是
      
       <context:annotation-config/>。请看下面的配置:

            <?xml version="1.0" encoding="UTF-8" ?>
            <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-2.5.xsd
             http://www.springframework.org/schema/context
             http://www.springframework.org/schema/context/spring-context-2.5.xsd">
             <context:annotation-config/>

            <bean id="chinese" class="com.Chinese"></bean>
            <bean id="American" class="com.American"></bean>
            <bean id="test" class="com.Test"></bean>

            </beans>
                     

         <context:annotationconfig/> 将隐式地向 Spring 容器注册 AutowiredAnnotationBeanPostProcessor、CommonAnnotationBeanPostProcessor、PersistenceAnnotationBeanPostProcessor 以及 equiredAnnotationBeanPostProcessor 这 4 个 BeanPostProcessor。


    15.4 使用 @Component或者@Component("bean名")    @Component默认bean名为类名  首字母小写

                然我们可以通过 @Autowired 或 @Resource 在 Bean 类中使用自动注入功能,但是 Bean 还是在 XML 文件中通过 <bean> 进行定义 —— 也就是说,在 XML 配置文件中定义 Bean,通过 @Autowired 或 @Resource 为 Bean 的成员变量、方法入参或构造函数入参提供自动注入的功能。能否也通过注释定义 Bean,从 XML 配置文件中完全移除 Bean 定义的配置呢?答案是肯定的,我们通过 Spring 2.5 提供的 @Component 注释就可以达到这个目标了。
   
     <?xml version="1.0" encoding="UTF-8" ?>
        <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-2.5.xsd
         http://www.springframework.org/schema/context
         http://www.springframework.org/schema/context/spring-context-2.5.xsd">
            <context:component-scan base-package="com"/><!--com包-->
        </beans>

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值