Spring

Spring的两个核心部分:Ioc 和AOP

(1)Ioc:控制反转,把创建对象过程交给Spring进行管理,降低耦合度

(2)AOP:面向切面,不修改源代码进行功能增强

Spring 特点:

(1)方便解耦,简化开发

(2)AOP编程支持

(3)方便程序测试

(4)方便和其他框架进行整合

(5)方便进行事务操作

(6)降低API开发难度

Ioc

Ioc底层原理

        xml解析、工厂模式、反射

第一步 xml配置文件,配置创建的对象

第二步 有service类和dao类,创建工厂类

  • xml解析
  • 通过反射创建对象

Ioc基于Ioc容器完成,Ioc容器底层就是对象工厂。

Spring提供Ioc容器实现两种方式:(两个接口)

(1)BeanFactory:Ioc容器基本实现,是Spring内部的使用接口,不提供开发人员进行使用

  • 加载配置文件时候不会创建对象,在获取对象(使用)才去创建对象

(2)ApplicationContext:BeanFactory接口的子接口,提供更多更强大的功能,一般由开发人员进行使用

  • 加载配置文件时候就会把在配置文件中的对象进行创建

Ioc操作Bean管理

Bean管理指的是两个操作:Spring创建对象、Spring注入属性

Bean管理操作有两种方式:

  • 基于xml配置文件方式实现
基于xml方式创建对象:
1.在 spring 配置文件中,使用 bean 标签,标签里面添加对应属性,就可以实现对象创
2.在 bean 标签有很多属性,常用的属性
* id 属性:唯一标识
* class 属性:类全路径(包类路径)
3.创建对象时候,默认也是执行无参数构造方法完成对象创建

 基于xml方式注入属性:

1.使用set方法进行注入

2.使用有参数构造进行注入

3.p名称空间注入

FactoryBean

Spring 有两种类型Bean,一种普通Bean,另外一种工厂Bean(FactoryBean)

普通Bean:在配置文件中定义Bean类型就是返回类型

工厂Bean:在配置文件张定义Bean类型可以和返回类型不一样

实现:创建类,让这个类作为工厂Bean,实现接口FactoryBean,重写其getObject方法,定义返回Bean的类型。

Bean作用域

在Spring里面,默认情况下,Bean是单例对象

1.在 spring 配置文件 bean 标签里面有属性( scope )用于设置单实例还是多实例
scope 属性值
  • 第一个值 默认值,singleton,表示是单实例对象
  • 第二个值 prototype,表示是多实例对象
2.singleton 和 prototype 区别
第一 singleton 单实例, prototype 多实例
第二 设置 scope 值是 singleton 时候,加载 spring 配置文件时候就会创建单实例对象
设置 scope 值是 prototype 时候,不是在加载 spring 配置文件时候创建 对象,在调用
getBean 方法时候创建多实例对象

 Bean生命周期

1.通过构造器创建 bean 实例(无参数构造)
2.为 bean 的属性设置值和对其他 bean 引用(调用 set 方法)
3.把 bean 实例传递 bean 后置处理器的方法 postProcessBeforeInitialization
4.调用 bean 的初始化的方法(需要进行配置初始化的方法)
5.把 bean 实例传递 bean 后置处理器的方法 postProcessAfterInitialization
6.bean 可以使用了(对象获取到了)
7.当容器关闭时候,调用 bean 的销毁的方法(需要进行配置销毁的方法)

xml自动装配

自动装配:根据指定装配规则(属性名称或者属性类型),Spring 自动将匹配的属性值进行注入

 实现自动装配

bean 标签属性 autowire ,配置自动装配
autowire 属性常用两个值:
  • byName 根据属性名称注入 ,注入值 bean 的 id 值和类属性名称一样
  • byType 根据属性类型注入

引入外部properties属性文件到Spring配置文件中

引入context名称空间

 基于注解方式

1 、什么是注解
注解是代码特殊标记,格式: @ 注解名称 ( 属性名称 = 属性值 , 属性名称 = 属性值 ..)
使用注解,注解作用在类上面,方法上面,属性上面
使用注解目的:简化 xml 配置
2 Spring 针对 Bean 管理中创建对象提供注解
@Component
@Service
@Controller
@Repository
* 上面四个注解功能是一样的,都可以用来创建 bean 实例
3 、基于注解方式实现对象创建
第一步 引入依赖
第二步 开启组件扫描
<!--开启组件扫描
1 如果扫描多个包,多个包使用逗号隔开
2 扫描包上层目录
-->
第三步 创建类,在类上面添加创建对象注解
4、开启组件扫描细节配置
use-default-filters="false" 表示现在不使用默认 filter,自己配置 filter
context:include-filter ,设置扫描哪些内容
下面配置扫描包所有内容
context:exclude-filter: 设置哪些内容不进行扫描
5 、基于注解方式实现属性注入
(1) @Autowired :根据 属性类型 进行自动装配
第一步 把 service 和 dao 对象创建,在 service 和 dao 类添加创建对象注解
第二步 在 service 注入 dao 对象,在 service 类添加 dao 类型属性,在属性上面使用注解
(2) @Qualifier :根据 名称 进行注入
这个@Qualifier 注解的使用,和@Autowired 一起使用
(3) @Resource :可以根据类型注入,可以根据名称注入
(4) @Value :注入普通类型属性

 AOP

AOP:面向切面编程,不通过修改源代码方式,在主干功能里面添加新功能。

底层原理

1.AOP底层使用动态代理

(1)两种情况代理

第一种 有接口情况,使用JDK动态代理

创建接口实现类代理对象,增强类的方法

第二种 没有接口情况 ,使用CGLIB动态代理

创建子类的代理对象,增强类的方法

AOP(JDK动态代理)

1.使用JDK动态代理,使用Proxy类里面的方法创建接口的实现类代理对象

(1)调用newProxyInstance方法

方法有三个参数:

  • 第一参数,类加载器
  • 第二参数,增强方法所在的类,这个类实现的接口,支持多个接口
  • 第三参数,实现这个接口InvocationHandler的类,用于创建代理对象,写增强的逻辑

 2.JDK动态代理实现代码

public class JDKProxy {
    public static void main(String[] args) {
        Class[] interfaces = {UserDao.class};
        UserDaoImpl userDao = new UserDaoImpl();
        // 创建接口实现类代理对象
//        Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new InvocationHandler() {
//            @Override
              // 匿名内部类
//            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//                return null;
//            }
//        })
        // 创建完代理对象要进行强转
        UserDao user= (UserDao)Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(userDao));
        int res = user.add(1,2);
        System.out.println("执行结果:"+res);
    }
}

class UserDaoProxy implements InvocationHandler{
    // 引入被代理的对象,UserDaoImpl
    private Object obj;
    public UserDaoProxy(Object obj){
        this.obj = obj;
    }
    // 写增强的逻辑
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // 方法执行前的逻辑
        System.out.println("方法之前执行。。。" + method.getName() + ":传递的参数为:"+ Arrays.toString(args));
        // 执行要增强的方法
        Object res = method.invoke(obj, args);
        // 方法执行后的逻辑
        System.out.println("方法之后执行。。。"+obj);
        return res;
    }
}

AOP(术语)

1.连接点

类里面可以被增强的方法被称为连接点

2.切入点

实际被真正增强的方法,称为切入点

3.通知(增强)

(1)实际增强的逻辑部分称为通知(增强)。如:权限判断等

(2)通知有多种类型

  • 前置通知(@Before())
  • 后置通知(@AfterReturning())
  • 环绕通知(@Around(),方法执行前后都会执行的增强)
  • 异常通知(@AfterThrowing())
  • 最终通知(@After(),方法有异常也会执行)

4.切面

是动作,把通知应用到切入点的过程

AOP操作

1.Spring框架中一般基于AspectJ实现AOP操作

AspectJ,AspectJ 不是 Spring 组成部分,独立 AOP 框架,一般把 AspectJ 和 Spirng 框架一起使用,进行 AOP 操作

2.基于AspectJ实现AOP操作

  • 基于 xml 配置文件实现
  • 基于注解方式实现(使用)

3.在项目工程里面引入AOP相关依赖

4.切入点表达式

(1)切入点表达式作用:知道对哪个类里面的哪个方法进行增强

(2)语法结构:

execution([权限修饰符][返回类型(可以省略不写)][类全路径][方法名称]([参数列表]))

举例:

对 com.wym.spring5.dao.BookDao类里面的add方法进行增强

  • execution(* com.wym.spring5.dao.BookDao.add(..))

对 com.wym.spring5.dao.BookDao类里面所有的方法进行增强

  • execution(* com.wym.spring5.dao.BookDao.*(..))

对 com.wym.spring5.dao包里面所有类,类里面所有的方法进行增强

  • execution(* com.wym.spring5.dao.*.*(..))

AOP操作(AspectJ注解) 

1.创建被增强的类

// 被增强类
public class User {

    public void add(){
        System.out.println("add....");
    }

}

2. 创建增强类

(1)在增强类里创建方法,让不同方法代表不同的通知类型

// 增强的类
public class UserProxy {
    // 前置通知
    public void before(){
        System.out.println("before....");
    }
}

3.进行通知的配置

(1)在Spring配置文件中,添加context命名空间,开启注解扫描

<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.xsd
                           http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
                           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
    <!--开启注解扫描-->
    <context:component-scan base-package="com.wym.spring5.aopanno"></context:component-scan>

(2)使用注解@Component创建User和UserProxy对象

// 被增强类
@Component
public class User {

    public void add(){
        System.out.println("add....");
    }

}

(3)在增强类上面添加注解@Aspect

// 增强的类
@Component
@Aspect // 生成代理对象
public class UserProxy {
    
    // 前置通知
    // @Before 注解表示作为前置通知
    @Before(value = "execution(* com.wym.spring5.aopanno.User.add(..))")
    public void before(){
        System.out.println("before....");
    }
    @Around(value = "execution(* com.wym.spring5.aopanno.User.add(..))")
    public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        System.out.println("环绕之前....");
        proceedingJoinPoint.proceed();
        System.out.println("环绕之后....");
    }
}

(4)在Spring配置文件中,增加aop命名空间,开启生成代理对象

    <!--开启Aspect生成代理对象-->
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>

4.配置不同类型的通知

(1)在增强类里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置

5.相同的切入点抽取

6.有多个增强类对同一个方法进行增强,设置增强类的优先级

(1)在增强类上面添加注解@Order(数字类型值),数字类型值越小优先级越高

// 增强的类
@Component
@Aspect // 生成代理对象
@Order(3)
public class UserProxy {
    // 相同切入点抽取
    @Pointcut(value = "execution(* com.wym.spring5.aopanno.User.add(..))")
    public void pointDemo(){

    }

    // 前置通知
    // @Before 注解表示作为前置通知
//    @Before(value = "execution(* com.wym.spring5.aopanno.User.add(..))")
    @Before(value = "pointDemo()")
    public void before(){
        System.out.println("before....");
    }
    @Around(value = "execution(* com.wym.spring5.aopanno.User.add(..))")
    public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        System.out.println("环绕之前....");
        proceedingJoinPoint.proceed();
        System.out.println("环绕之后....");
    }
}

7.完全使用注解开发

(1)创建配置类,不需要创建xml配置文件

@Configuration
@ComponentScan
@EnableAspectJAutoProxy(proxyTargetClass = true)
public class ConfigAop {
}

 

AOP操作(AspectJ配置文件)

1、创建两个类,增强类和被增强类,创建方法

2、在 spring 配置文件中创建两个类对象

3、在 spring 配置文件中配置切入点

        *配置 aop 增强 

                *配置切入点
                *配置切面

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值