Spring系列框架的学习-基础篇

6 篇文章 0 订阅
5 篇文章 0 订阅

Spring框架学习

一、Spring基础

1.1 最基础最核心

IOC(控制反转):代码中的创建对象,赋值等由代码控制,反转交由容器(Spring是一种容器)控制。
AOP(面向切面),实现模块之间、类之间的解耦合。
Spring容器:容器中存放JAVA对象,由容器进行管理。
Spring:使用的是依赖注入技术,底层使用的是反射机制。

1.2 优点

  • 轻量:非常小,没有依赖
  • IOC:针对接口编程,减少对象之间的耦合(能够实现业务对象之间的解耦合,例:service与dao对象之间的解耦合)
  • AOP:支持AOP编程,将应用程序业务逻辑与系统服务分开。
  • 组合框架:容易跟其它的框架一起使用

1.3 Spring的引入

在pom.xml中引入

    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.3.20</version>
    </dependency>

1.4 Spring简单实例

由XML文件实现JAVA对象的注入,然后通过读取Spring容器对象,通过key获取指定的对象。默认同一个名称的对象只存在一个(即单例模式)。

实例
  • 创建接口
public interface StudyTest {
    void oneTest();
}
  • 接口实现
public class StudyTestImpl implements StudyTest {
    @Override
    public void oneTest() {
        System.out.println("调用了StudyTestImpl 的oneTest方法!");
    }
}
  • Spring配置
<?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.xsd">
     <!--
        beans里存放了需要Spring创建的对象。一个bean对应一个创建的对象。
        所有的对象都会存放在Spring的特定Map中,id的值相当于key,class的对象相当于value。Spring相当于执行了map.put("【id的值】",【class的对象】);
        StudyTest studyTest = new StudyTestImpl(); map.put("studyTest",studyTest);
      -->
    
    <bean id="studyTest" class="com.xqj.test.impl.StudyTestImpl" />
</beans>
  • 调用
    AplicationContext是一个接口;实现类有FileSystemXmlApplicationContext(从本地磁盘中找寻配置文件的类)和ClassPathXmlApplicationContext(从项目汇总找配置文件)
public class AppTest{
    @Test
    public void test1(){
        String config = "SpringConfig.xml";
        //创建Spring的容器ApplicationContext时,会创建配置中所有的对象。
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext(config);
        //从Spring的对象Map中读取指定对象(bean中的ID)
        StudyTest studyTest = (StudyTest) applicationContext.getBean("studyTest");
        studyTest.oneTest();
    }
}

二、Spring学习

2.1 依赖注入(di)

  • 开发人员在项目中只需要提供对象的名称,由容器实现对象的创建、查找、赋值。
  • 可以在配置文件中对简单数据类型(JAVA的基本数据类型和String类型)进行注入。
  • 在创建Spring容器对象时,配置文件的路径默认是以src->main->resources为根目录读取。
2.1.1 注入分类
  • 1.set注入(设值注入):Spring调用类的set方法进行属性赋值。
    1)简单类型的set注入:property name=“属性名” value=“属性值”
    2)引用类型的set注入:property name=“属性名” ref=“引用类型的bean的id”
  • 2.构造注入:Spring调用有参构造方法(constructor-arg name=“” value=“” index=“”)
    1)name:表示构造方法的形参名
    2)index:表示构造方法的参数位置,从0开始
  • 3.自动注入:由Spring根据一些规则(常用的byName,byType)给引用类型完成赋值
    1)byName:按名称注入,java类中引用类型的属性名和Spring容器中bean的id一样,数据类型一样
    2)byType:按类型注入,java类中引用类型的是数据类型和Spring容器中bean的class同源关系(父子关系)
2.1.2 set注入
1. 简单数据类型的set注入
<property name="属性名" value="属性的值"/>
实例
  • 创建实体类
public class User {
    private String name;
    private int age;

    public void setName(String name){
        this.name = name;
    }
    public void setAge(int age){
        this.age = age;
    }
        
    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
  • 配置文件依赖注入
<bean id="myUser" class="com.xqj.test.pojo.User">
     <property name="name" value="张三"></property>
     <property name="age" value="16"></property>
 </bean>
  • 调用
    @Test
    public void test2(){
        String config = "SpringConfig.xml";
        //Spring的对象ApplicationContext
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext(config);
        User user = (User) applicationContext.getBean("myUser");
        System.out.println("注入:"+user);
    }
  • 使用Spring注入,在配置文件中给实体类赋值时,需要实体类中包含set方法。
  • 只要类中存在set方法,就可以通过Spring配置注入。
2. 引用类型的set注入
<property name="属性名" ref="引用的bean的ID"/>
实例
  • 创建实体类
//公司
public class Company {
    private String name;
    private String address;

    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getAddress() {
        return address;
    }
    public void setAddress(String address) {
        this.address = address;
    }
    @Override
    public String toString() {
        return "Company{" +
                "name='" + name + '\'' +
                ", address='" + address + '\'' +
                '}';
    }
}
//员工
public class Worker {
    private String name;
    private int age;
    private Company company;

    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public Company getCompany() {
        return company;
    }
    public void setCompany(Company company) {
        this.company = company;
    }

    @Override
    public String toString() {
        return "Worker{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", company=" + company +
                '}';
    }
  • 配置
    <bean id="worker" class="com.xqj.test.pojo.Worker">
        <property name="name" value="张三"></property>
        <property name="age" value="22"></property>
        <property name="company" ref="company"></property>
    </bean>
    <bean id="company" class="com.xqj.test.pojo.Company">
        <property name="name" value="张三"></property>
        <property name="address" value="广州"></property>
    </bean>
  • 调用
    @Test
    public void test3(){
        String config = "SpringConfig.xml";
        //Spring的对象ApplicationContext
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext(config);
        Worker worker = (Worker) applicationContext.getBean("worker");
        System.out.println("注入:"+worker);
    }
2.1.3 构造注入
实例
  • 实体类
//在Worker类中添加有参构造函数
  public Worker(String myName,int myAge,Company myCompany){
        this.name=myName;
        this.age=myAge;
        this.company=myCompany;
    }
  • 配置
    <bean id="company" class="com.xqj.test.pojo.Company">
        <property name="name" value="张三"></property>
        <property name="address" value="广州"></property>
    </bean>

    <bean id="myWorker" class="com.xqj.test.pojo.Worker">
        <!--  name:有参构造函数的参数名;value:参数的值;index:对应参数的位置(name和index其中一个就可以定位到参数)  -->
        <constructor-arg name="myName" value="张三" index="0"/>
        <constructor-arg name="myAge" value="26" index="1"/>
        <constructor-arg name="myCompany" ref="company" index="2"/>
    </bean>
  • 调用
    @Test
    public void test3(){
        String config = "SpringConfig.xml";
        //Spring的对象ApplicationContext
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext(config);
        //通过bean的ID获取指定的对象
        Worker worker = (Worker) applicationContext.getBean("myWorker");
        System.out.println("注入:"+worker);
    }
2.1.4 引用类型的自动注入

Spring框架根据某些规则可以给引用类型赋值,不用再给引用类型赋值,使用的规则常用的是byName和byType

1.byName方式自动注入
  • byName(按名称注入):JAVA代码中对象的名称需要与配置文件中bean的ID保持一致,才能实现引用注入。
实例

Worker 实体类的Company 对象定义的名称与配置文件中注入的Company对象bean的ID一致。

  • 实体类
public class Worker {
    private String name;
    private int age;
    private Company company;

    public Worker(){}

    public Worker(String myName,int myAge,Company myCompany){
        this.name=myName;
        this.age=myAge;
        this.company=myCompany;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public Company getCompany() {
        return company;
    }
    public void setCompany(Company company) {
        this.company = company;
    }
    @Override
    public String toString() {
        return "Worker{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", company=" + company +
                '}';
    }
}
  • 配置文件
//定义autowire属性为byName,且引用的类型名称与引用类型的bean的ID一致。
   <bean id="worker" class="com.xqj.test.test01.pojo.Worker"  autowire="byName">
        <property name="name" value="张三"></property>
        <property name="age" value="22"></property>
    </bean>
    <bean id="company" class="com.xqj.test.test01.pojo.Company">
        <property name="name" value="XX有限公司"></property>
        <property name="address" value="广州"></property>
    </bean>
2. byType方式自动注入

bean中定义autowire为byType。引用的类型会根据bean属性class的值(引用类型的地址)找到响应的bean对象。
但是可能会出现不唯一的情况,容易报错。

实例

2.2 注解

使用步骤
  • 1.加入依赖:spring-context,间接加入spring-aop
  • 2.在类中加入注解
  • 3.在spring的配置文件中,加入组件扫描器的标签<context:component-scan base-package=“包名”/>
2.2.1 常用注解
1.列表
注解名注解作用
@Component创建对象
@Service创建Service对象,处理业务逻辑,可以有事务功能
@Repository创建dao对象,用来访问数据库的
@Controller创建控制器对象的,接收请求,显示处理结果的
@ResourceJDK(1.6以上)中的注解,使用自动注入给引用类型赋值,支持byName、byType,默认byName
@AutowiredSpring框架中引用类型的赋值注解,支持byName、byType,默认byType
@Value简单数据类型的赋值
2.描述
  • @Component
作用:注解该类为实体类。等同于配置文件的bean功能。
用法:
在定义的类上面添加@Component(value = "")或@Component(""),value是该类的名称,相当于配置文件中的bean标签的id属性。value的值必须是唯一的。或@Component写法,标签为所注解的类首字母小写的名称。
其它能创建对象的注解(以下三个注解除了跟Component一样能创建对象,还有其它功能):
@Repository(持久层)放在dao的实现类上面,表示创建dao对象(能访问数据库)。
@Service(业务层)放在service的实现类上面,创建service对象(service对象是做业务层处理,可以有事务等功能)。
@Controller(控制器上面)放在控制器(处理器)上面,创建控制器对象(控制器对象能够接收用户提交的请求和最终的处理结果)
  • @Value
作用:简单数据类型的赋值(Value是String数据类型)。
用法:@Value(value="")或者@Value("")。在定义的属性上方使用,无需set方法。也可以用在set方法之上。
  • @Autowired
作用:Spring框架提供的注解,实现引用类的赋值。使用自动注入的原理,支持byName、byType。默认byType自动注入。
byType的方式: @Autowired,在定义的属性上方使用,无需set方法。也可以用在set方法之上。
byName的方式: @Autowired+@Qualifier("")一起使用。Qualifier是引用对象Component注解定义的名称或者bean引入是的id。
Autowired的required属性:boolean类型,默认为true。
required=true引用类型赋值失败,则报错,并终止程序。
required=false引用类型赋值失败,则程序不报错,继续运行,引用对象为null。
  • @Resource
作用:来自JDK1.6以上的注解,Spring框架提供了关于这个注解的功能支持。可以通过它实现对引用类型的赋值。是通过自动注入的原理,支持byName、byType,默认byName。在执行的时候会先按byName的方式执行,执行失败后,会使用byType执行。
用法:
@Resource。在引用类型的上方使用,无需set方法。也可以在set方法上方使用。
@Resource(name=""),只使用byName的方式对引用类型进行赋值。
2.2.2 组件扫描器
  • 配置文件标签:
使用规范:<context:component-scan base-package="扫描指定路径下的所有文件"/>
base-package:指定注解在项目中的包名
工作方式:Spring会扫描遍历base-package指定的包,把包中和子包中的所有类,找到类中的注解,按照注解的功能创建对象,或给属性赋值。
2.2.3 JavaConfig类
1.1 JavaConfig类
  • JavaConfig:使用java类作为xml配置文件的替代。是配置Spring容器的纯java的方式,在这个java类中可以创建java对象,把对象放入Spring容器中。
  • 使用两个注解
    1)@Configuration:放在类的上方,表示该类是作为配置文件的java类使用。
    2)@Bean:声明对象,把对象注入Spring容器中。
  • 实例:
//实体类
public class Student {
    private String name;
    private int age;
    private String sex;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public String getSex() {
        return sex;
    }
    public void setSex(String sex) {
        this.sex = sex;
    }
    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", sex=" + sex +
                '}';
    }

//JavaConfig的java类
@Configuration
public class MyConfig {
    /**
     * @Bean:默认方法名
     *      属性 name:指定自定义名称(bean的名称,相当于配置文件bean标签的id)
     * @return
     */
    @Bean(name="getStudent")
    public Student getStudent(){
        Student student = new Student();
        student.setName("张三");
        student.setAge(18);
        student.setSex("男");
        return student;
    }
}

//单元测试类
public class MyTest01 {
    @Test
    public void test01(){
        ApplicationContext applicationContext = new AnnotationConfigApplicationContext(MyConfig.class);
        Student student = (Student) applicationContext.getBean("getStudent");
        System.out.println(student);
    }
1.2 @ImportResource
  • @ImportResource:用于导入其它的xml配置文件

  • 使用:
    @Configuration
    @ImportResource(value = “classpath:test01/applicationContext.xml”)
    public class MyConfig {}

  • 实例:

//实体类
public class User {
    private String userName;
    private String passWord;
    private int number;
    public String getUserName() {
        return userName;
    }
    public void setUserName(String userName) {
        this.userName = userName;
    }
    public String getPassWord() {
        return passWord;
    }
    public void setPassWord(String passWord) {
        this.passWord = passWord;
    }
    public int getNumber() {
        return number;
    }
    public void setNumber(int number) {
        this.number = number;
    }
    @Override
    public String toString() {
        return "User{" +
                "userName='" + userName + '\'' +
                ", passWord='" + passWord + '\'' +
                ", number=" + number +
                '}';
    }
}

//配置文件
    <bean id="myUser" class="com.xqj.test01.User">
        <property name="userName" value="abc123"/>
        <property name="passWord" value="xqj123"/>
        <property name="number" value="1234567"/>
    </bean>

//JavaConfig的java类(通过@ImportResource引入配置文件)
//多个配置文件 @ImportResource(value = "{classpath:test01/applicationContext.xml,xxxxxx.xml}")
@Configuration
@ImportResource(value = "classpath:test01/applicationContext.xml")
public class MyConfig {

}

//单元测试
    @Test
    public void test02(){
        ApplicationContext applicationContext = new AnnotationConfigApplicationContext(MyConfig.class);
        User user = (User) applicationContext.getBean("myUser");
        System.out.println(user);
    }
1.3 @PropertyResource
  • @PropertyResource:读取属性配置文件
  • 使用:在resources目录下创建config.properties文件,在使用的时候,通过${属性的key}取值
  • 实例:
//实体类
@Component("tiger")
public class Tiger {
    @Value("${tiger.name}")
    private String name;
    @Value("${tiger.age}")
    private int age;
}

//配置文件(config.properties)
tiger.name=东北虎
tiger.age=3

//JavaConfig的java类(通过@PropertySource引入自定义文件)
@PropertySource(value="classpath:test01/config.properties") 
@ComponentScan(basePackages = "com.xqj.test01.pojo")  //将目录下的所有类创建对象,并交给Spring容器
public class MyConfig {}

//单元测试
   @Test
    public void test03(){
        ApplicationContext applicationContext = new AnnotationConfigApplicationContext(MyConfig.class);
        //实体类@Component("tiger")中定义的名称
        Tiger tiger= (Tiger) applicationContext.getBean("tiger");
        System.out.println(tiger);
    }

2.3 AOP面向切面编程

  • AOP(面向切面编程):
    1.面向切面编程是从动态角度考虑程序运行过程。可通过运行期动态代理实现程序功能的统一维护的一种技术。
    2.AOP就是动态代理的规范化,把动态代理的实现步骤、方式都定义好了,让开发人员用一种统一的方式实现动态代理。
    3.AOP的底层就是采用动态代理模式实现的。采用两种方式代理,JDK的动态代理和CGLIB的动态代理。

  • 理解面向切面编程:
    1)需要在分析项目功能时,找出切面(标识增强的功能,非业务功能。比如:日志、事务等)。
    2)合理的安排切面的执行时间(在目标方法前还是方法后)。
    3)合理的安全的切面执行位置,在哪个类,哪个方法增加增强功能。

  • AOP开发三要素:
    1)切面的功能代码(切面是做什么功能的)。
    2)切面能执行的位置(pointcut切入点,多个连接点的集合。是对目标对象的多个方法进行增强)。
    3)切面执行的时间(Active通知表示时间,是在目标方法之前还是之后)。

  • 应用场景:
    1)需要给类中的方法进行补充功能。原有的类功能不全,由没有源代码的情况下,可以通过AOP进行功能增强。
    2)需要个多个类增加相同的功能,可以通过AOP实现。
    3)需要给业务方法增加日志、事务等非业务的功能。

2.3.1 动态代理
  • 动态代理:是指程序在整个运行过程中根本就不存在目标的代理类,目标对象的代理对象只是由代理生成工具(不是真实定义的类)在程序运行时由JVM根据反射等机制动态生成的。代理对象与目标对象的代理关系在程序运行时才确立。
  • 作用:1)在目标类源代码不改变的情况下增加功能。2)减少代码的重复。3)专注业务逻辑代码。4)解耦合,将业务代码与非业务代码(日志、事务等)分离。
  • 方式:
  • 1.Spring:Spring在内部实现了AOP的规范,能做AOP的工作。主要在事务处理时使用。但是过于笨重,很少使用。
  • 2.aspectJ:一个开源的专门做AOP的框架,Spring框架中集成了aspectJ框架,可以通过Spring使用aspectJ的功能。
    1)使用xml的配置文件实现。
    2)使用注解,有5个注解。
1.JDK动态代理

使用:通过JDK中的Proxy、Method、InvocationHandler创建代理对象。
注意事项:JDK的动态需要目标对象必须实现接口(java设计上的需求),如果目标类不存在接口,则无法实现。
步骤:

  • 1.创建目标类
  • 2.创建InvocationHandler接口的实现类,在这个类实现给目标增强的功能
  • 3.使用JDK中的类Proxy,创建代理对象,实现创建对象的能力
2.CGLIB动态代理

原理:生成目标类的子类,在子类中重写父类的方法实现增强,这个子类对象就是代理对象。
注意事项:目标必须是能够被继承,即不能是final的类和方法。
优点:CGLIB的代理效率高于JDK,经常应用在框架中(Spring、Hibernate)等。

3.动态代理实例
AspectJ表达式

AspectJ定义了专门的表达式用于指定切入点。

  • 表达式的原型是:
    execution(modifiers-pattern? ret-type-pattern declaring-type-pattern?name-pattern(param-pattern) throws-pattern?)
  • 表达式的四个部分:
    execution(访问权限 方法返回值 方法声明(参数) 异常类型)
  • 表达式描述
    ? 表示可选的部分,没有则表示为必填的参数。
名称描述
modifiers-pattern访问权限类型,比如public、private等
ret-type-pattern返回值类型
declaring-type-pattern包名类名
name-pattern(param-pattern)方法名(参数类型和参数个数)
throws-pattern抛出异常类型
  • 符号意义
符号意义
*0至多个任意字符
用在方法参数中,表示任意多个参数;用在包名后,表示当前包及其子包路径
+用在类名后,表示当前类及其子类;用在接口后,表示当前接口及其实现类
  • 表达式举例
例子说明
execution(public * *(…))指定切入点为:任意公共方法。
execution(* set*(…))指定切入点为:任何一个方法名以set开始的方法
execution(* com.xyz.service..(…))指定切入点为:定义在service包中的任意类的任意方法。不包含子包中的类
execution(* com.xyz.service….(…))指定切入点为:定义在service包或者子包中的任意类的任意方法
execution(* …service*.*(…))指定切入点为:所有包下的service子包下所有类(接口)中的所有方法
动态代理实例

前置通知@Before

  • service业务类
//AspectJ框架是基于JDK的动态代理,所以切面对象必须是接口的实现类。
public interface SomeService {
    void doSome(String name,Integer age);
}

@Service("someService")
public class SomeServiceImpl implements SomeService {
    @Override
    public void doSome(String name,Integer age) {
        //需要在这个目标中进行增强

        System.out.println("---SomeServiceImpl的doSome方法---");
    }
}
  • 增强类(aspectJ)
@Aspect
@Component
public class MyAspect {
    /**
     * 方法的定义要求:
     * 1.必须是公共方法(public)
     * 2.方法没有返回值(void)
     * 3.方法名称自定义
     * 4.方法可以没有参数,也可以有参数。如果有参数,则参数不可自定义,需要遵循参数规范。
     */

    /**
     * @Before 前置通知注解
     * 属性value:是切入点表达式,表示切面功能执行的位置(目标方法的前面)
     *
     */
    @Before(value = "execution(public void com.xqj.aop1.service.impl.SomeServiceImpl.doSome(String,Integer))")
    public void doBefore(){
        System.out.println("doBefore增强---方法前执行");
    }
  • 配置类
    <!-- <bean id="someService" class="com.xqj.aop1.service.impl.SomeServiceImpl"/>
    <bean id="myAspect" class="com.xqj.aop1.function.aop.MyAspect"/> -->

    <context:component-scan base-package="com.xqj.aop1"/>
    <!--  spectj-autoproxy:会把Spring容器中的所有的目标对象一次性都生成代理对象  -->
    <aop:aspectj-autoproxy/>
  • 测试类
@Test
    public void aopTest1(){
        String congfig = "aop01/applicationContext.xml";
        ApplicationContext applicationContext = (ApplicationContext)new ClassPathXmlApplicationContext(congfig);
        SomeService someService = (SomeService)applicationContext.getBean("someService");
        someService.doSome("杰克",12);
    }
JoinPoint通知

JoinPoint:业务方法,要加入切面的业务方法。
作用:可以在通知中方法中获取方法执行的信息,例如:方法名,方法的实参。
JoinPoint参数的值是由框架赋予,必须是第一个位置的参数。

  • 前置通知@Before
@Before(value = "execution(public void com.xqj.aop1.service.impl.SomeServiceImpl.doSome(String,Integer))")
public void doBefore1(JoinPoint jp){
        System.out.println("方法的签名:"+jp.getSignature());
        System.out.println("方法的名称:"+jp.getSignature().getName());
        Object[] objs = jp.getArgs();
        for(Object obj : objs){
            System.out.println("参数的值:"+obj);
        }
        System.out.println("doBefore增强---方法前执行");
    }
  • 后置通知:@AfterReturning
    属性:
    •  1.value 切入点表达式(execution())
      
    •  2.returning 自定义的变量,表示目标方法的返回值。自定义变量名必须和通知方法的形参名一致。
      
    特点:
    •  1.在目标方法之后执行
      
    •  2.能够获取到目标方法的返回值,可以根据这个返回值做不同的处理功能。
      
    •  3.可以修改这个返回值。
      
    实例
//定义带有返回值的接口,并实现。 impl实现类代码
    @Override
    public String doOther(String name, Integer age) {
        System.out.println("---SomeServiceImpl的doOther方法---");
        return "doOther";
    }

//添加后置通知。切面类中添加增强方法。
    @AfterReturning(value = "execution(* *..SomeServiceImpl.doOther(String,Integer))",returning = "obj")
    public void doAfter1(JoinPoint jp,Object obj){
        System.out.println("JoinPoint后置通知的方法的定义:"+jp.getSignature());
        System.out.println("目标方法doSome之后执行增强方法");
        System.out.println("doAfter1后置通知的returning参数:"+obj);
    }
  • 环绕通知:@Around
    属性:value 切入点表达式
    特点:

    • 1.功能最强的通知
    • 2.在目标方法的前和后都能增强功能
    • 3.控制目标方法是否被调用执行
    • 4.修改原来的目标方法的执行结果,影响最后的调用结果。
      环绕通知经常做事务,在目标方法之前开启事务,执行目标方法,在目标方法之后提交事务。

    环绕通知等同于JDK动态代理的InvocationHandler接口。参数ProceedingJoinPoint等同于Method。返回值就是目标方法的执行结果,可以被修改。
    实例:

	//定义带有返回值的接口,并实现。 impl实现类代码
    @Override
    public String doEncircle(String name, Integer age) {
        System.out.println("---SomeServiceImpl的doEncircle方法---");
        return "doEncircle";
    }
    
	//添加环绕通知。切面类中添加增强方法。
    @Around(value = "execution(* *..SomeServiceImpl.doEncircle(String,Integer))")
    public Object doEncircle(ProceedingJoinPoint pjp) throws Throwable {
        Object[] objs = pjp.getArgs();
        String name = "";
        if(null != objs && objs.length > 0){
            name = (String)objs[0];
        }
        System.out.println("环绕通知:目标方法之前执行的方法--获取当前时间-"+new Date());
        Object result = null;
        if("汤姆".equals(name)){
            result = pjp.proceed();
        }
        System.out.println("环绕通知:目标方法之后执行的方法--执行某些事物");
        return result;
    }
    @Around(value = "execution(* *..SomeServiceImpl.doEncircle(String,Integer))")
    public Object doEncircle(ProceedingJoinPoint pjp) throws Throwable {
    	//获取目标方法的参数
        Object[] objs = pjp.getArgs();
        String name = "";
        if(null != objs && objs.length > 0){
            name = (String)objs[0];
        }
        //增强内容
        System.out.println("环绕通知:目标方法之前执行的方法--获取当前时间-"+new Date());
        Object result = null;
        //不符合条件,则业务的方法不被执行
        if("汤姆".equals(name)){
            //执行目标的业务方法
            result = pjp.proceed();
        }
        //增强内容
        System.out.println("环绕通知:目标方法之后执行的方法--执行某些事物");
        return result+"1111";
    }

	//调用
    @Test
    public void aopTest3(){
        String congfig = "aop01/applicationContext.xml";
        ApplicationContext applicationContext = (ApplicationContext)new ClassPathXmlApplicationContext(congfig);
        SomeService someService = (SomeService)applicationContext.getBean("someService");
        //相当于直接调用MyAspect的doEncircle方法。MyAspect的doEncircle返回什么,rel等于什么 
        String rel = someService.doEncircle("汤姆",33);
        System.out.println("返回值:"+rel);
    }
  • 异常通知:@AfterThrowing
    定义格式:
    •   1.public
      
    •   2.没有返回值
      
    •   3.方法名称自定义
      
    •   4.方法有一个Exception,如果有就是JoinPoint
      
    属性:
    •   1.value 切入点
      
    •   2.throwing 自定义的变量,表示目标方法抛出的异常对象。变量名必须和方法的形参名一致
      
    特点:
    •   1.在目标方法抛出异常时执行
      
    •   2.可以做异常的监控程序,监控目标方法执行时是不是有异常。有异常可以通过发短信、发邮件进行通知。
      

效果类似try{}catch(Exception e){},在执行异常时执行异常通知。

  • 最终通知:@After
    属性:
    • 1.value 切入点表达式
      特点:
    • 1.总是会被执行
    • 2.在目标方法之后执行
      效果类似try{}catch(Exception e){}finally{},在finally中执行。
Pointcut管理切入点
  • @Pointcut:定义和管理切入点。如果项目中有多个切入点表达式是重复或是可以服用的,都可以使用@Pointcut。
  • 属性:value 切入点表达式
  • 位置:在自定义方法的上方
  • 特点:当使用 @Pointcut定义在一个方法的上面,此时这个方法的名称就是这个切入点表达式的别名。其它通知使用时,value属性使用该方法别名,即可代替切入点表达式。
  • 实例:
    @Before(value = "expression()")
    public void function1(){
        System.out.println("执行方法function1");
    }

    @After("expression()")
    public void function2(){
        System.out.println("执行方法function2");
    }

    @Pointcut("execution(* *..SomeServiceImpl.doEncircle(String,Integer))")
    public void expression(){
    }
CGLIB方式
  • 目标类没有接口,使用CGLIB动态代理,Spring框架会自定应用cglib。
  • 实例:
//切面类
@Aspect
@Component
public class MyAspect2 {
    @Before(value = "expression()")
    public void function1(){
        System.out.println("执行方法function1");
    }

    @After("expression()")
    public void function2(){
        System.out.println("执行方法function2");
    }

    @Pointcut("execution(public void com.xqj.aop1.service.MyService.doSome())")
    public void expression(){
    }
}

//执行类
  @Test
  public void aopTest4(){
      String congfig = "aop01/applicationContext.xml";
      ApplicationContext applicationContext = (ApplicationContext)new ClassPathXmlApplicationContext(congfig);
      MyService myService = (MyService)applicationContext.getBean("myService");
      myService.doSome();
  }
  • 如果有接口,但是不用JDK动态代理,而是用CGLIB动态代理,则需要修改配置文件。
	<!--    <aop:aspectj-autoproxy/>-->
    <aop:aspectj-autoproxy proxy-target-class="true"/>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值