JAVA之Spring学习(第二天)

23 篇文章 0 订阅
15 篇文章 0 订阅

一,学习大纲

动态代理设计模式 (JDK cglib)
AOP 详解
AOP 中几种通知类型
AOP 两种实现方式 (Schema-base AspectJ)

二,AOP

        1.AOP:中文名称面向切面编程
        2.英文名称 :( Aspect Oriented Programming )
        3.正常程序执行流程都是纵向执行流程
                3.1 又叫面向切面编程, 在原有纵向执行流程中添加横切面
                3.2不需要修改原有程序代码
                        3.2.1高扩展性
                        3.2.2原有功能相当于释放了部分逻辑.让职责更加明确
        4.面向切面编程是什么 ?
                4.1 在程序原有纵向执行流程中, 针对某一个或某一些方法添加通
, 形成横切面过程就叫做面向切面编程 .
        5.常用概念
                5.1 原有功能: 切点 , pointcut
                5.2前置通知: 在切点之前执行的功能 . before advice
                5.3后置通知: 在切点之后执行的功能 ,after advice
                5.4如果切点执行过程中出现异常,会触发异常通知 .throws advice
                5.5所有功能总称叫做切面.
                5.6织入: 把切面嵌入到原有功能的过程叫做织入
        6.spring 提供了 2 AOP 实现方式
                6.1 Schema-based
                        6.1.1每个通知都需要实现接口或类
                        6.1.2配置 spring 配置文件时在<aop:config> 配置 6.2 AspectJ
                        6.2.1每个通知不需要实现接口或类
                        6.2.2配置 spring 配置文件是在<aop:config> 的子标签 <aop:aspect> 中配置

. Schema-based 实现步骤

 

        2.新建通知类
                2.1 新建前置通知类
                        2.1.1 arg0: 切点方法对象 Method 对象
                        2.1.2 arg1: 切点方法参数
                        2.1.3 arg2:切点在哪个对象中
public class MyBeforeAdvice implements MethodBeforeAdvice {

	@Override
	public void before(Method arg0, Object[] arg1, Object arg2) throws Throwable {
		System.out.println("我是前置通知");
		
	}

}
        2.2 新建后置通知类
                2.2.1 arg0: 切点方法返回值
                2.2.2 arg1:切点方法对象
                2.2.3 arg2:切点方法参数
                2.2.4 arg3:切点方法所在类的对象
public class MyAfterAdvice implements AfterReturningAdvice {

	@Override
	public void afterReturning(Object arg0, Method arg1, Object[] arg2, Object arg3) throws Throwable {
		System.out.println("执行后置通知");
		
	}

}
        3. 配置 spring 配置文件
                3.1 引入 aop 命名空间
                3.2配置通知类的<bean>
                3.3配置切面
                3.4 * 通配符,匹配任意方法名 , 任意类名 , 任意一级包名
                3.5如果希望匹配任意方法参数 (..)

<?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:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsdss">
        
        
<!--    <bean id="after" class="com.ljx.dome.MyAfterAdvice"></bean>
        <bean id="before" class="com.ljx.dome.MyBeforeAdvice"></bean>
        <aop:config>
          <aop:pointcut expression="execution(* com.ljx.dome.Demo.demo2())"  id="mypoint"/>
          <aop:advisor advice-ref="after" pointcut-ref="mypoint"/>
          <aop:advisor advice-ref="before" pointcut-ref="mypoint"/>
        </aop:config> -->
        
        <!--使用AspectJ配置异常处理  -->
        <bean id="myThrow" class="com.ljx.dome.Throw"></bean>
<!--         <aop:config>
           <aop:aspect ref="myThrow">
               配置切点 
               <aop:pointcut expression="execution(* com.ljx.dome.Demo.demo2())" id="mypoint"/>
               <aop:after-throwing method="throwAfter" pointcut-ref="mypoint"/>
           </aop:aspect>
        </aop:config> -->
        
        <bean id="demo" class="com.ljx.dome.Demo"></bean>
</beans>

. 配置异常通知的步骤(AspectJ 方式)

        1.只有当切点报异常才能触发异常通知
        2.在 spring 中有 AspectJ 方式提供了异常通知的办法 .
                2.1 如果希望通过 schema-base 实现需要按照特定的要求自己编
写方法 .
        3. 实现步骤 :
                3.1 新建类, 在类写任意名称的方法
public class MyThrowAdvice{ 
    public void myexception(Exception e1){
         System.out.println("执行异常通知 "+e1.getMessage());
 }
}
        3.2 在 spring 配置文件中配置
                3.2.1 <aop:aspect>的 ref 属性表示 : 方法在哪个类中 .
                3.2.2 <aop: xxxx/> 表示什么通知
                3.2.3 method: 当触发这个通知时, 调用哪个方法
                3.2.4 throwing: 异常对象名, 必须和通知中方法参数名相同 (
以不在通知中声明异常对象 )
<bean id="mythrow" class="com.bjsxt.advice.MyThrowAdvice"></bean> 
   <aop:config> 
      <aop:aspect ref="mythrow">
        <aop:pointcut expression="execution(* com.bjsxt.test.Demo.demo1())" id="mypoint"/>
       <aop:after-throwing method="myexception" pointcut-ref="mypoint" throwing="e1"/>
      </aop:aspect>
 </aop:config> 
<bean id="demo" class="com.bjsxt.test.Demo"></bean>

五.异常通知(Schema-based 方式)

        1. 新建一个类实现 throwsAdvice 接口
                1.1必须自己写方法,且必须叫 afterThrowing
                1.2有两种参数方式
                        1.2.1 必须是 1 个或 4
                1.3 异常类型要与切点报的异常类型一致
public class MyThrow implements ThrowsAdvice{ 
        public void afterThrowing(Exception ex) throws Throwable { 
                 System.out.println("执行异常通过-schema-base 方式 "); 
}
 }
        2. 在 ApplicationContext.xml 配置
<bean id="mythrow" class="com.bjsxt.advice.MyThrow"></bean>
      <aop:config>
            <aop:pointcut expression="execution(* com.bjsxt.test.Demo.demo1())" id="mypoint"/> 
            <aop:advisor advice-ref="mythrow" pointcut-ref="mypoint" />
      </aop:config>
 <bean id="demo" class="com.bjsxt.test.Demo"></bean>

.使用 AspectJ 方式实现

        1. 新建类 , 不用实现
                1.1 类中方法名任意
package com.ljx.text;

import org.aopalliance.intercept.MethodInvocation;
import org.aspectj.lang.ProceedingJoinPoint;

public class Befort {
    public void Befort01(String name,int age) {
    	System.out.println(name+"     "+age);
    	System.out.println("前置");
    }
    
    public void after01() {
    	System.out.println("后置");
    }
    public void thorw() {
    	System.out.println("异常处理");
    }
    public void befort() {
    	System.out.println("前置");
    }
    public Object around( ProceedingJoinPoint m) throws Throwable {
    	System.out.println("环绕前置");
    	Object proceed = m.proceed();
    	System.out.println("环绕后置");
		return proceed;
    	
    }
}

        1.2 配置 spring 配置文件
                1.2.1 <aop:after/> 后置通知, 是否出现异常都执行
                1.2.2 <aop:after-returing/> 后置通知 , 只有当切点正确执行时 执行
                1.2.3<aop:after/> 和 <aop:after-returing/> <aop:after-throwing/> 执行顺序和配置顺序有关
                1.2.4 execution() 括号不能扩上 args
                1.2.5 中间使用 and 不能使用 && spring and 解析成 &&
                1.2.6 args(名称) 名称自定义的 . 顺序和 demo1( 参数 , 参数 ) 对应
                1.2.7 <aop:before/> arg-names=” 名 称 ” 名 称 来 源 于 expression=”” args(), 名称必须一样
                        1.2.7.1 args() 有几个参数,arg-names 里面必须有几个参数
                        1.2.7.2 arg-names=”” 里面名称必须和通知方法参数名对应
 <bean id="befort" class="com.ljx.text.Befort"></bean>
       <aop:config>
             <aop:aspect ref="befort">
                <!--  <aop:pointcut expression="execution(* com.ljx.dome.Demo.demo2())" id="mypoint"/> -->
                 <aop:pointcut expression="execution(* com.ljx.dome.Demo.demo1(String,int)) and args(name,age)" id="mypoint"/>
                 <aop:before method="Befort01" pointcut-ref="mypoint" arg-names="name,age"/>
                 
                 <aop:pointcut expression="execution(* com.ljx.dome.Demo.demo2())" id="mypoint1"/>
                 <aop:before method="befort" pointcut-ref="mypoint1"/>
                 <aop:after-returning method="after01" pointcut-ref="mypoint1"/>
                 <aop:after-throwing method="thorw" pointcut-ref="mypoint1"/>
                 <aop:around method="around" pointcut-ref="mypoint1"/>
             </aop:aspect>
       </aop:config>
       <bean id="demo" class="com.ljx.dome.Demo"></bean>

. 使用注解(基于 Aspect)

        1. spring 不会自动去寻找注解 , 必须告诉 spring 哪些包下的类中可能 有注解
                1.1 引入 xmlns:context
 <context:component-scan base-package="com.ljx.text,com.ljx.dome"></context:component-scan>
        <aop:aspectj-autoproxy proxy-target-class="true"></aop:aspectj-autoproxy>
        2.@Component
                2.1相当于<bean/>
                2.2如果没有参数,把类名首字母变小写 , 相当于 <bean id=””/>
                2.3 @Component(“自定义名称”)
        3.实现步骤:
                3.1 在 spring 配置文件中设置注解在哪些包中
                3.2 在 Demo 类中添加 @Componet
                        3.2.1 在方法上添加@Pointcut(“”) 定义切点
@Component
public class Demo {
	@Pointcut("execution(* com.ljx.dome.Demo.demo1())")
  public void demo1() {
		//int i=1/0;
	  System.out.println("demo1");
  }
}
                3.3 在通知类中配置
                        3.3.1 @Component 类被 spring 管理
                        3.3.2 @Aspect 相当于<aop:aspect/>表示通知方法在当前类中
@Component
@Aspect
public class Advice {
	@Before("com.ljx.dome.Demo.demo1()")
   public void befort() {
	   System.out.println("我是注解前置");
   }
	@After("com.ljx.dome.Demo.demo1()")
	public void after() {
	   System.out.println("我是注解后置");
	}
	@AfterThrowing("com.ljx.dome.Demo.demo1()")
	public void error() {
		System.out.println("我是注解异常");
	}
}
(经典面试题)

.代理设计模式

        1.设计模式: 前人总结的一套解决特定问题的代码 .
        2.代理设计模式优点:
                2.1保护真实对象
                2.2让真实对象职责更明确.
                2.3扩展
        3.代理设计模式
                3.1真实对象.(老总 )
                3.2 代理对象( 秘书 )
                3.3抽象对象(抽象功能 ), 谈小目标

. 静态代理设计模式

        1. 由代理对象代理所有真实对象的功能 .
                1.1 自己编写代理类
                1.2 每个代理的功能需要单独编写
        2. 静态代理设计模式的缺点 :
                2.1 当代理功能比较多时, 代理类中方法需要写很多 .

. 动态代理

        1.为了解决静态代理频繁编写代理功能缺点.
        2.分类:
                2.1 JDK 提供的
                2.2 cglib 动态代理

十一. JDK 动态代理

        1.和 cglib 动态代理对比
                1.1优点:jdk 自带 , 不需要额外导入 jar
                1.2缺点:
                        1.2.1 真实对象必须实现接口
                        1.2.2利用反射机制.效率不高.
        2.使用 JDK 动态代理时可能出现下面异常

                 2.1 出现原因:希望把接口对象转换为具体真实对象

十二: cglib 动态代理

        1. cglib 优点 :
                1.1 基于字节码, 生成真实对象的子类 .
                        1.1.1 运行效率高于 JDK 动态代理 .
                1.2 不需要实现接口
        2.cglib 缺点 :
                2.1 非 JDK 功能 , 需要额外导入 jar
        3.使用 spring aop , 只要出现 Proxy 和真实对象转换异常
                3.1设置为 true 使用 cglib
                3.2设置为 false 使用 jdk( 默认值
<aop:aspectj-autoproxy proxy-target-class="true"></aop:aspectj-autoproxy>

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值