spring5学习笔记

Spring5框架

* Spring5框架概述

  • Spring 是轻量级的开源的 JavaEE 框架
  • Spring 可以解决企业应用开发的复杂性
  • Spring 有两个核心部分:IOC 和 Aop
    • IOC控制反转,把创建对象过程交给 Spring 进行管理
    • Aop面向切面,不修改源代码进行功能增强
  • Spring 特点
    • 方便解耦,简化开发
    • Aop 编程支持
    • 方便程序测试
    • 方便和其他框架进行整合
    • 方便进行事务操作
    • 降低 API 开发难度

* IOC概念和原理

- 什么是IOC

  • 控制反转,把对象创建和对象之间的调用过程,交给 Spring 进行管理
  • 使用 IOC 目的:为了耦合度降低
  • 做入门案例就是 IOC 实现

- IOC底层原理

xml 解析、工厂模式、反射

在这里插入图片描述

- BeanFactory 接口

在这里插入图片描述

* IOC操作Bean管理

  • 什么是 Bean 管理

    • Bean 管理指的是两个操作
    • Spring 创建对象
    • Spirng 注入属性
  • Bean 管理操作有两种方式

    • 基于 xml 配置文件方式实现
    • 基于注解方式实现

* IOC操作bean管理(基于xml)

- 基于XML创建对象

<bean id="Book" class="spring5_demo.Book">
  • 在 spring 配置文件中,使用 bean 标签,标签里面添加对应属性,就可以实现对象创建

  • 在 bean 标签有很多属性,介绍常用的属性

    • id 属性:唯一标识
    • class 属性:类全路径(包类路径)
  • 创建对象时候,默认也是执行无参数构造方法完成对象创建

- 基于XML注入属性

DI:依赖注入,就是注入属性

+ 注入方式一:set方法注入

步骤:

  • 创建类,定义属性和对应的 set 方法
    在这里插入图片描述

  • 在 spring 配置文件配置对象创建,配置属性注入

在这里插入图片描述

+ 注入方式二:使用有参数构造进行注入

步骤:

  • 创建类,定义属性,创建属性对应有参数构造方法
    在这里插入图片描述

  • 在 spring 配置文件中进行配置

注意:使用构造函数进行注入时默认调用的是无参构造器,因为我们重写了构造器,所以覆盖了无参构造器,所以在还没有写constructor-arg等时会报错,只有通过constructor-arg这个标签对每一个属性进行赋值才不会报错

在这里插入图片描述

+ 注入方式三:p名称空间注入(了解)

下方第一步中即为在配置文件中加上灰色的那一行文字:xmlns:p=“http://www.springframework.org/schema/p”

第二部即为直接在class后方添加属性p:bname等

在这里插入图片描述

* 注入属性

- 注入空值和特殊符号

+ set方法注入空值

通过set方法注入空值:

<property name="address">
    <null/>
</property>

即为在property标签内加上

+ set方法注入特殊符号

首先在value=""内大于号小于号的转义仍然可用,即为:

小于号为&lt;
大于号为&gt;

如果不想转义还要写入特殊符号,就需要如下:

要传入的value值为 <<南京>>
就需要如下的写法:<![CDATA[    ]]>
< property  name= "address">
< value><![CDATA[<<南京>>]]></ value>
</ property>

- 注入属性-外部Bean

步骤:

  • 创建两个类 service 类和 dao 类
  • 在 service 调用 dao 里面的方法
  • 在 spring 配置文件中进行配置

在这里插入图片描述

- 注入属性-内部Bean

以下是以员工和部门进行举例

(1)一对多关系:部门和员工
一个部门有多个员工,一个员工属于一个部门
部门是一,员工是多
(2)在实体类之间表示一对多关系,员工表示所属部门,使用对象类型属性进行表示

在这里插入图片描述在这里插入图片描述

- 注入属性-级联赋值

+ 写法1

在这里插入图片描述

+ 写法2

在这里插入图片描述
在这里插入图片描述

* IOC操作Bean管理(XML注入集合属性)

- 注入数组类型属性

<!--        数组类型注入-->
        <property name="courses">
<!--            在下方可用list标签或者array标签-->
            <array>
            <value>java课程</value>
            <value>javaWeb课程</value>
            <value>mysql课程</value>
            <value>Linux课程</value>
            </array>
        </property>

- 注入List类型属性

<!--        List类型注入-->
        <property name="list">
            <list>
                <value>张三</value>
                <value>李四</value>
                <value>王五</value>
                <value>小桑</value>
            </list>
        </property>

- 注入Map类型属性

<!--        Map类型注入-->
        <property name="maps">
            <map>
                <entry key="Java" value="java"/>
                <entry key="C++" value="c++"/>
                <entry key="JavaWeb" value="jw"/>
                <entry key="Mysql" value="mysql"/>
            </map>
        </property>

- 注入Set类型属性

<!--        set类型注入-->
        <property name="set">
            <set>
                <value>java</value>
                <value>javaWeb</value>
                <value>mysql</value>
                <value>c++</value>
            </set>
        </property>

- 在集合中注入对象类型值

<!--        要注入List集合类型,其中的值为对象-->
        <property name="courseList">
            <list>
                <ref bean="course1"/>
                <ref bean="course2"/>
                <ref bean="course3"/>
            </list>
        </property>
    </bean>
    <bean id="course1" class="collectiontype.Course">
        <property name="cname" value="课程1"/>
    </bean>
    <bean id="course2" class="collectiontype.Course">
        <property name="cname" value="课程1"/>
    </bean>
    <bean id="course3" class="collectiontype.Course">
        <property name="cname" value="课程1"/>
    </bean>

- 将集合注入部分提取出来

+ 在 spring 配置文件中引入名称空间 util

在这里插入图片描述

+ 使用 util 标签完成 list 集合注入提取
<util:list id="bookList">
    <value>哇咔咔</value>
    <value>呜啦啦</value>
    <value>欸嘿嘿</value>
</util:list>
<bean id="book" class="collectiontype.Book">
    <property name="list" ref="bookList">
    </property>
</bean>

- FactoryBean

  • Spring 有两种类型 bean ,一种普通 bean ,另外一种工厂 bean (FactoryBean)
  • 普通 bean :在配置文件中定义 bean
  • 工厂 bean :在配置文件定义 bean类型可以和返回类型不一样
    • 第一步 创建类,让这个类作为工厂 bean,实现接口 FactoryBean
    • 第二步 实现接口里面的方法,在实现的方法中定义返回的 bean 类型
//定义的类型为MyBean类型,返回的为Course类型
public class MyBean implements FactoryBean<Course> {
    //定义返回bean
    @Override
    public Course getObject() throws Exception {
        Course course = new Course();
        course.setCname("abc");
        return course;
    }
}

要注意,如果要返回与定义类型不同的类型时,需要注意修改Course course = context.getBean(“myBean”, Course.class);

//测试代码
public void test(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean7.xml");
        Course course = context.getBean("myBean", Course.class);
        System.out.println(course);
    }

- bean作用域

+ bean默认为单实例

Spring中默认情况下bean为单实例对象

在这里插入图片描述

+ 设置单或多实例(scope属性)
  • 在 spring 配置文件 bean 标签里面有属性(scope)用于设置单实例还是多实例

  • scope 属性值

    • 第一个值 默认值,singleton,表示是单实例对象

    • 第二个值 prototype,表示是多实例对象

在这里插入图片描述

  • singleton 和 prototype 区别
    • 第一 singleton 单实例,prototype 多实例
    • 第二 设置 scope 值是 singleton 时候,加载 spring 配置文件时候就会创建单实例对象设置 scope 值是 prototype 时候,不是在加载 spring 配置文件时候创建 对象,在调用getBean 方法时候创建多实例对象

- bean生命周期

生命周期即为从对象创建到销毁的过程

+ 生命周期
  • 通过构造器创建 bean 实例(无参数构造)
  • 为 bean 的属性设置值和对其他 bean 引用(调用 set 方法)
  • 调用 bean 的初始化的方法(需要进行配置初始化的方法)
  • bean 可以使用了(对象获取到了)
  • 当容器关闭时候,调用 bean 的销毁的方法(需要进行配置销毁的方法)

在这里插入图片描述

代码演示:

//设置的实例对象
public class Orders {
    private String oname;
    public Orders() {
        System.out.println("第一步调用Orders内的无参构造器");
    }
    public void setOname(String oname) {
        this.oname = oname;
        System.out.println("第二部调用oname的set方法设置属性的值");
    }
    //创建执行初始化方法
    public void initMethod(){
        System.out.println("第三步执行初始化方法");
    }
    //销毁方法
    public void destroyMethod(){
        System.out.println("第五步执行销毁的方法");
    }
}
//测试代码
public void test(){
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("bean8.xml");
        Orders orders = context.getBean("orders",Orders.class);
        System.out.println("第四步获取创建bean实例对象");
        System.out.println(orders);
        //销毁bean实例对象,调用close方法
        orders.destroyMethod();
        context.close();
    }

- bean的后置处理器(七步生命周期)

  • 通过构造器创建 bean 实例(无参数构造)

  • 为 bean 的属性设置值和对其他 bean 引用(调用 set 方法)

  • 把 n bean 实例传递 n bean 后置处理器的方法 postProcessBeforeInitialization

  • 调用 bean 的初始化的方法(需要进行配置初始化的方法)

  • 把 bean 实例传递 bean 后置处理器的方法 postProcessAfterInitialization

  • bean 可以使用了(对象获取到了)

  • 当容器关闭时候,调用 bean 的销毁的方法(需要进行配置销毁的方法)
    在这里插入图片描述

- 演示添加后置处理器效果

创建类,实现接口 BeanPostProcessor,创建后置处理器

//后置处理器在初始化前后要执行的代码
public class MyBeanPost implements BeanPostProcessor {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("插入在第二部和第三步之间 :在初始化开始之前执行的方法");
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("插入在第三部和第四步之间 :在初始化结束之前执行的方法");
        return bean;
    }
}
<!--    配置后置处理器-->
    <bean id="myBeanPost" class="bean.MyBeanPost"/>

- XML自动装配

  • 手动装配即为手动设置哪个属性有什么值,如在标签property中手动装配name属性和对于的value值

  • 自动装配不需要写property这一行,根据指定装配规则(属性名称或者属性类型),Spring 自动将匹配的属性值进行注入

+ 自动装配分为byName与byType

bean标签属性autowire,配置自动装配

autowire属性常用两个值,一个是byName,即为按照属性名称注入,一个是byType,即为按照属性类型注入

代码演示:

//根据属性名称自动装配
<bean id="emp" class="Spring_learning_package.autoWire.Emp" autowire="byName">
<property name="dept" ref="dept"/>
</bean>
<bean id="dept" class="Spring_learning_package.autoWire.Dept"></bean>
//根据属性类型自动装配
<bean id="emp" class="Spring_learning_package.autoWire.Emp" autowire="byType">
<property name="dept" ref="dept"/>
</bean>
<bean id="dept" class="Spring_learning_package.autoWire.Dept"></bean>

* 外部属性文件

<!--    引入-->
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:p="http://www.springframework.org/schema/p"
       xmlns:util="http://www.springframework.org/schema/util"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/util
http://www.springframework.org/schema/util/spring-util.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">

- 直接配置数据库信息

  • 配置德鲁伊连接池
  • 引入德鲁伊连接池依赖 jar
<!--    最基本的配置方式直接配置连接池-->
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
        <property name="url" value="jdbc:mysql://localhost:3306/userDb"/>
        <property name="username" value="root"/>
        <property name="password" value="hd200308041410"/>
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
    </bean>

- 引入外部属性文件配置数据库连接池

  • 创建外部属性文件,properties 格式文件,写数据库信息
  • 把外部 properties 属性文件引入到 spring 配置文件中,引入 context 名称空间
<!--    引入外部属性文件-->
    <context:property-placeholder location="jdbc.properties"/>
<!--    配置连接池-->
    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
        <property name="driverClassName" value="${prop.driverClass}"/>
        <property name="url" value="${prop.url}"/>
        <property name="username" value="${prop.userName}"/>
        <property name="password" value="${prop.password}"/>
    </bean>

* 注解(@)

- 什么是注解

  • 注解是代码特殊标记,格式:

    @注解名称(属性名称=属性值, 属性名称=属性值…)

  • 使用注解,注解作用在类上面,方法上面,属性上面

  • 使用注解目的:简化 xml 配置

- Spring针对Bean管理中创建对象提供注解

  • @Component
  • @Service
  • @Controller
  • @Repository

上面四个注解功能相同,都可以创建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:p="http://www.springframework.org/schema/p"
           xmlns:context="http://www.springframework.org/schema/context"
           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">
    
  • 开启组件扫描

    <!--    开启组件扫描-->
        <!--    扫描多个包的写法,即为多个包的路径使用逗号隔开-->
        <!--    如果要扫描的包都在一个包下,那么可以扫描上层的包-->
        <context:component-scan base-package="Spring_learning_package"/>
    
  • 创建类,在类上添加创建对象注解

//Component后跟上的value可以不写,如果不写的话值默认为类名称的首字母小写
@Component(value = "userService") //实际上与<bean id="userService" class="XXXXXX>
public class UserService {
    public void add(){
        System.out.println("service add.....");
    }
}

- 开启组件扫描细节配置

  • use-default-filters=“false” 表示现在不使用默认 filter,自己配置 filter
  • context:include-filter ,设置扫描哪些内容
<context:component-scan base-package="Spring_learning_package" use-default-filters="false">
        <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>
  • 下面配置扫描包所有内容
  • context:exclude-filter: 设置哪些内容不进行扫描
<context:component-scan base-package="Spring_learning_package">
        <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

- 基于注解方式实现属性注入

+ @Autowired 根据属性类型自动装配
  • 第一步 把 service 和 dao 对象创建,在 service 和 dao 类添加创建对象注解
  • 第二步 在 service 注入 dao 对象,在 service 类添加 dao 类型属性,在属性上面使用注解
//Component后跟上的value可以不写,如果不写的话值默认为类名称的首字母小写
@Service
public class UserService {
    @Autowired
    private UserDao userDao;
    publi,c void add(){
        System.out.println("service add.....");
        userDao.add();
    }
}
+ @Qualifier:根据名称进行注入

这个@Qualifier 注解的使用,和上面@Autowired 一起使用

如下案例,其中@Qualifier(value = “userDaoImpl1”)中的值要与@Repository(value = “userDaoImpl1”)相同才能匹配上

@Service
public class UserService {
    @Autowired
    //通过名称进行注入
    @Qualifier(value = "userDaoImpl1")
    private UserDao userDao;
    public void add(){
        System.out.println("service add.....");
        userDao.add();
    }
}
@Repository(value = "userDaoImpl1")
public class UserDaoImpl implements UserDao{
    @Override
    public void add() {
        System.out.println("dao add.....");
    }
}
+ @Resource 根据类型注入或名称注入

通过该标签指定实现类,指定依据可以自行选择,但是需要在两个类中均相同

//    根据类型注入或名称注入
    @Resource(name = "userDaoImpl1")
    private UserDao userDao;
+ @Value:注入普通类型属性

可以在该标签的下方直接设置一个值,默认标签内的value值会赋值给该设置的值

注意:只能传入一个String值,但是会自动转化为你需要的类型,如int,char,boolean等,但是传入的数据要复合类型值规范,否则会报错

@Value(value = "abcde")
    private String name;
@Value(value = "3")
    private int name;
@Value(value = "a")
    private char name;
@Value(value = "true")
    private boolean name;

- 完全注解开发

  • 创建配置类,替代 xml 配置文件

    @Configuration//将当前类作为一个配置类
    //下面一行与bean文件中的<context:component-scan base-package="Spring_learning_package"/>是等价的
    @ComponentScan(basePackages = "Spring_learning_package")
    public class SpringConfig {
        
    }
    
  • 编写测试类(除了上方的修改外只有测试类不同其他不变)

    //使用完全注解开发时的测试方法
        @Test
        public void testService2() {
            ApplicationContext context
                    =  new AnnotationConfigApplicationContext(SpringConfig. class);
            UserService userService = context.getBean( "userService",
                    UserService. class);
            System. out .println(userService);
            userService.add();
        }
    

* AOP概念

  • AOP即为面向切面编程(方面),利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得
    业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
  • 通俗描述:不通过修改源代码方式,在主干功能里面添加新功能
  • 使用登录例子说明 AOP
    在这里插入图片描述

* AOP底层原理

- AOP使用动态代理

AOP有两种动态代理模式

+ 有接口采用JDK动态代理

创建接口实现类代理对象,增强类的方法
在这里插入图片描述

+ 无接口采用CGLIB动态代理

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

在这里插入图片描述

- AOP(JDK动态代理)

使用 JDK 动态代理,使用 Proxy 类里面的方法创建代理对象

+ 调用 newProxyInstance 方法

在这里插入图片描述

方法有三个参数:

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

- 编写JDK动态代理代码

  • 创建接口,定义方法

    public interface UserDao2 {
        public int add(int a,int b);
        public String update(String id);
    }
    
  • 创建接口实现类,实现方法

    public class UserDaoImpl2 implements UserDao2{
        @Override
        public int add(int a, int b) {
            return a+b;
        }
        @Override
        public String update(String id) {
            return id;
        }
    }
    
  • 使用 Proxy 类创建接口代理对象

public class JDKProxy {
    public static void main(String[] args) {
        Class[] interfaces = {UserDao2.class};
//        Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new InvocationHandler() {
//            @Override
//            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//                return null;
//            }
//        });
        UserDaoImpl2 userDaoImpl2 = new UserDaoImpl2();
        UserDao2 userDao2 = (UserDao2) Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces,new UserDaoProxy(userDaoImpl2));
        int result = userDao2.add(1,2);
        System.out.println("result:"+result);
    }
}
//创建代理对象代码
class UserDaoProxy implements InvocationHandler{
    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术语

+ 连接点

在这里插入图片描述

+ 切入点

在这里插入图片描述

+ 通知(增强)

在这里插入图片描述在这里插入图片描述

+ 切面

在这里插入图片描述

- AOP操作

  • Spring 框架一般都是基于 AspectJ 实现 AOP 操作
    • AspectJ 不是 Spring 组成部分,独立 AOP 框架,一般把 AspectJ 和 Spirng 框架一起使用,进行 AOP 操作
  • 基于 AspectJ 实现 AOP 操作
    • 基于 xml 配置文件实现
    • 基于注解方式实现(使用)
  • 在项目工程里面引入 AOP
    在这里插入图片描述

- 切入点表达式

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

  • 语法结构:

    execution([权限修饰符] [返回类型] [类全路径] [方法名称] ( [ 参数列表 ] ) )
    
//举例 1:对 com.atguigu.dao.BookDao 类里面的 add 进行增强
execution(* com.atguigu.dao.BookDao.add(..))
    
//举例 2:对 com.atguigu.dao.BookDao 类里面的所有的方法进行增强
execution(* com.atguigu.dao.BookDao.* (..))
    
//举例 3:对 com.atguigu.dao 包里面所有类,类里面所有方法进行增强
execution(* com.atguigu.dao.*.* (..))

- AOP 操作(AspectJ 注解)

  • 在增强类中创建方法

在这里插入图片描述

  • 在 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"
       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">
  • 使用注解创建 User 和 UserProxy 对象
    在这里插入图片描述

  • 在增强类上面添加注解 @Aspect

    //增强的类
    @Component
    @Aspect //生成代理对象
    public class UserProxy {}
    
  • 在 spring 配置文件中开启生成代理对象

    <!--    开启AspectJ,生成代理对象-->
        <aop:aspectj-autoproxy/>
    
  • 配置不同类型的通知

    //增强的类
    @Component
    @Aspect //生成代理对象
    public class UserProxy {
        //前置通知
    //@Before 注解表示作为前置通知
        @Before(value = "execution(* Spring_learning_package.aopanno.User.add(..))")
        public void before() {
            System.out.println("before.........");
        }
    
        //后置通知(返回通知)
        @AfterReturning(value = "execution(* Spring_learning_package.aopanno.User.add(..))")
        public void afterReturning() {
            System.out.println("afterReturning.........");
        }
    
        //最终通知
        @After(value = "execution(* Spring_learning_package.aopanno.User.add(..))")
        public void after() {
            System.out.println("after.........");
        }
    
        //异常通知
        @AfterThrowing(value = "execution(* Spring_learning_package.aopanno.User.add(..))")
        public void afterThrowing() {
            System.out.println("afterThrowing.........");
        }
    
        //环绕通知
        @Around(value = "execution(* Spring_learning_package.aopanno.User.add(..))")
        public void around(ProceedingJoinPoint proceedingJoinPoint) throws
                Throwable {
            System.out.println("环绕之前 .........");
    //被增强的方法执行
            proceedingJoinPoint.proceed();
            System.out.println("环绕之后 .........");
        }
    }
    
  • 测试输出

在这里插入图片描述

+ 增强类优先级

如果有多个增强类对同一个方法进行增强,设置增强类优先级

注意:设置优先级需要使用到@Order注解,在其后括号内填入从1开始得到数字,数字越小优先级越高

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

+ 完全注解开发
@Configuration
@ComponentScan(basePackages = { "com.atguigu"})
@EnableAspectJAutoProxy(proxyTargetClass =  true)
public class ConfigAop {
}

- AOP 操作(AspectJ 配置文件)

  • 创建两个类,增强类和被增强类,创建方法
  • 在 spring 配置文件中创建两个类对象
<!--    创建对象-->
    <bean id="book" class="Spring_learning_package.aopT.Book"/>
    <bean id="bookProxy" class="Spring_learning_package.aopT.BookProxy"/>
  • 在 spring 配置文件中配置切入点
<!--    配置aop增强-->
    <aop:config>
<!--        切入点-->
        <aop:pointcut id="p" expression="execution(* Spring_learning_package.aopT.Book.buy(..))"/>
<!--        配置切面-->
        <aop:aspect ref="bookProxy">
<!--            增强作用在具体的方法上-->
            <aop:before method="before" pointcut-ref="p"/>
        </aop:aspect>
    </aop:config>

* JdbcTemplate对数据库增删改查

- 什么是JdbcTemplate

Spring 框架对 JDBC 进行封装,使用JdbeTemplate 方便实现对数据库操作

- 进行对数据库表的添加操作

  • 配置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"
           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="Spring_learning_package"/>
        <!-- 数据库连接池 -->
        <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
              destroy-method="close">
            <property name="url" value="jdbc:mysql:///user_db?serverTimezone=UTC" />
            <property name="username" value="root" />
            <property name="password" value="hd200308041410" />
            <property name="driverClassName" value="com.mysql.jdbc.Driver" />
        </bean>
        <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
            <property name="dataSource" ref="dataSource"/>
        </bean>
    </beans>
    
  • 创建实体类
    在这里插入图片描述

  • 编写Dao,BookDaoImpl

    @Repository
    public class BookDaoImpl implements BookDao{
    @Autowired
        private JdbcTemplate jdbcTemplate;
    
        @Override
        public void add(Book book) {
            String sql = "insert into t_book values(?,?,?)";
    
            int update = jdbcTemplate.update(sql,book.getUserId(),book.getUsername(),book.getUstatus());
            System.out.println(update);
        }
    }
    
    @Service
    public class BookService {
    @Autowired
        private BookDao bookDao;
    public void addBook(Book book){
    bookDao.add(book);
    }
    }
    
    • 编写测试类

      public class TestBook {
          @Test
          public void testJdbcTemplate(){
              ApplicationContext context = new ClassPathXmlApplicationContext("mysqlTest.xml");
              BookService bookService = context.getBean("bookService",BookService.class);
              Book book = new Book();
              book.setUserId("123");
              book.setUsername("老港吗");
              book.setUstatus("1241wdqw12e");
              bookService.addBook(book);
          }
      }
      
      
    • 检查输出,输出为1,说明添加成功


      在这里插入图片描述

- 查询表中数据条数

public int selectCount() {
        String sql = "select count(*) from t_book";
        return jdbcTemplate.queryForObject(sql,Integer.class);
    }
 public int findCount(){
        return bookDao.selectCount();
    }
//查询返回的一个数
        int count = bookService.findCount();
        System.out.println(count);

- 查询指定id的对象数据

调用 **jdbcTemplate.queryForObject()**方法实现查询对象

//接口中编写查询方法
Book findBookInfo(String id);
//实现方法
@Override
    public Book findBookInfo(String id) {
        String sql = "select * from t_book where user_id=?";
        return jdbcTemplate.queryForObject(sql,new BeanPropertyRowMapper<Book>(Book.class),id);
//创建service内方法
public List<Book> findAll(){
        return bookDao.findAllBook();
    }
//测试
//返回指定id的对象
        Book book = bookService.findOne("1");
        System.out.println(book);

- JdbcTemplate批量操作mysql数据库

在这里插入图片描述

+ 批量添加
//接口中编写查询方法
void batchAddBook(List<Object[]> batchArgs);
//实现方法
@Override
    public void batchAddBook(List<Object[]> batchArgs) {
        String sql = "insert into t_book values(?,?,?)";
        int[] ints = jdbcTemplate.batchUpdate(sql,batchArgs);
        System.out.println(Arrays.toString(ints));
    }
List<Object[]> batchArgs = new ArrayList<>();
        Object[] o1 = {"3","java","dasdq"};
        Object[] o2 = {"4","vaskdv","aver"};
        Object[] o3 = {"5","1231dq","dcase"};
        batchArgs.add(o1);
        batchArgs.add(o2);
        batchArgs.add(o3);
        bookService.batchAdd(batchArgs);
//创建service内方法
public void batchAdd(List<Object[]> batchArgs){
        bookDao.batchAddBook(batchArgs);
    }
+ 批量修改
//接口方法
void batchUpdateBook(List<Object[]> batchArgs);
//实现接口方法
@Override
    public void batchUpdateBook(List<Object[]> batchArgs) {
        String sql = "update t_book set username=?,ustatus=? where user_id=?";
        int[] ints = jdbcTemplate.batchUpdate(sql,batchArgs);
        System.out.println(Arrays.toString(ints));
    }
//service类内方法
public void batchUpdate(List<Object[]> batchArgs){
        bookDao.batchUpdateBook(batchArgs);
    }
//测试方法
List<Object[]> batchArgs = new ArrayList<>();
        Object[] o1 = {"java","dasdq","3"};
        Object[] o2 = {"vaskdv","aver","4"};
        Object[] o3 = {"1231dq","dcase","5"};
        batchArgs.add(o1);
        batchArgs.add(o2);
        batchArgs.add(o3);
        bookService.batchUpdate(batchArgs);
+ 批量删除
//接口方法
void batchDeleteBook(List<Object[]> batchArgs);
//实现接口方法
@Override
    public void batchDeleteBook(List<Object[]> batchArgs) {
        String sql = "delete from t_book where user_id=?";
        int[] ints = jdbcTemplate.batchUpdate(sql,batchArgs);
        System.out.println(Arrays.toString(ints));
    }
//service类内方法
public void batchDelete(List<Object[]> batchArgs){
        bookDao.batchDeleteBook(batchArgs);
    }
//测试方法
List<Object[]> batchArgs = new ArrayList<>();
        Object[] o1 = {"3"};
        Object[] o2 = {"4"};
        Object[] o3 = {"5"};
        batchArgs.add(o1);
        batchArgs.add(o2);
        batchArgs.add(o3);
        bookService.batchDelete(batchArgs);

* 事务

事务是数据库操作最基本单元,逻辑上一组操作,要么都成功,如果有一个失败所有操
作都失败

- 事务四个特性(ACID)

  • 原子性
  • 一致性
  • 隔离性
  • 持久性

- 事务例子:银行转账

//xml配置
<?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.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="Spring_learning_package"/>
    <!-- 数据库连接池 -->
    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
          destroy-method="close">
        <property name="url" value="jdbc:mysql:///user_db?serverTimezone=UTC" />
        <property name="username" value="root" />
        <property name="password" value="hd200308041410" />
        <property name="driverClassName" value="com.mysql.jdbc.Driver" />
    </bean>
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"/>
    </bean>
</beans>
//接口
public interface UserDao1 {
    void addMoney();
    void reduceMoney();
}
//接口实现类
@Repository
public class UserDao1Impl1 implements UserDao1 {
@Autowired
    private JdbcTemplate jdbcTemplate;

    @Override
    public void reduceMoney() {
        String sql = "update t_account set money=money-? where username=?";
        jdbcTemplate.update(sql, 100, "lucy");

    }

    @Override
    public void addMoney() {
        String sql = "update t_account set money=money+? where username=?";
        jdbcTemplate.update(sql,100,"mary");
    }
}
//service类
@Service
public class UserService1 {
    @Autowired
    private UserDao1 userDao1;

    public void accountMoney() {
        userDao1.reduceMoney();
        userDao1.addMoney();
    }
}

注意,测试类中getBean方法获取的类不能与其他类的获取冲突,即不能获取同一个类

//测试类
public class TestBook {
    @Test
    public void testAccount(){
        ApplicationContext context = new ClassPathXmlApplicationContext("ShiWu.xml");
        UserService1 userService = context.getBean("userService1", UserService1.class);
        userService.accountMoney();
    }
}

- 异常情况

如果上方的代码遇到异常,就会导致可能一部分代码成功执行,剩下的没有成功执行,就会形成数据错误,因此需要在service类内进行一次异常检测与捕捉

在这里插入图片描述

- Spring事务操作介绍

  • 事务添加到 JavaEE 三层结构里面 Service 层(业务逻辑层)
  • 在 Spring 进行事务管理操作
    • 有两种方式:编程式事务管理和声明式事务管理(一般使用声明式管理)
  • 声明式事务管理
    • 基于注解方式(使用)
    • 基于 xml 配置文件方式
  • 在 Spring 进行声明式事务管理,底层使用 AOP 原理
  • Spring 事务管理 API
    • 提供一个接口,代表事务管理器,这个接口针对不同的框架提供不同的实现类

- 声明式事务操作参数配置

+ 注解@Transactional

在 service 类上面添加注解@Transactional ,在这个注解里面可以配置事务相关参数

在这里插入图片描述

+ propagation 事务传播行为

多事务方法直接进行调用

在这里插入图片描述

+ 七大事务传播行为及其传播属性

在这里插入图片描述

+ ioslation :事务隔离级别
  • 事务有特性成为隔离性,多事务操作之间不会产生影响。不考虑隔离性产生很多问题
  • 有三个读问题:脏读、不可重复读、虚(幻)读
  • 脏读:一个未提交事务读取到另一个未提交事务的数据

在这里插入图片描述

  • 不可重复读:一个未提交事务读取到另一提交事务修改数据

在这里插入图片描述

  • 虚读:一个未提交事务读取到另一提交事务添加数据

  • 通过设置事务隔离级别解决读的问题
    在这里插入图片描述

    在这里插入图片描述

+ timeout :超时时间
  • 事务需要在一定时间内进行提交,如果不提交进行回滚
  • 默认值是 -1 ,设置时间以秒为单位进行计算
  • 需要在 @Transactional 内添加 timeout 属性,在其后添加属性值
+ readOnly:是否只读
  • 读:查询操作,写:添加修改删除操作
  • readOnly 默认值 false,表示可以查询,可以添加修改删除操作
  • 设置 readOnly 值是 true,设置成 true 之后,只能查询
+ rollbackFor :回滚

设置出现哪些异常进行事务回滚

+ noRollbackFor :不回滚

设置出现哪些异常不进行事务回滚

- XML 声明式事务管理

  • 在spring 配置文件中进行配置
    第一步 配置事务管理器
    第二步 配置通知
    第三步 配置切入点和切面
<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.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 http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
    <!--    开启组件扫描-->
    <context:component-scan base-package="Spring_learning_package"/>
    <!-- 数据库连接池 -->
    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
          destroy-method="close">
        <property name="url" value="jdbc:mysql:///user_db?serverTimezone=UTC"/>
        <property name="username" value="root"/>
        <property name="password" value="hd200308041410"/>
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
    </bean>
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    <!--    创建事务管理器-->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <!--        注入数据源-->
        <property name="dataSource" ref="dataSource"/>
    </bean>
<!--    配置通知-->
    <tx:advice id = "txadvice">
        <tx:attributes>
<!--            指定在哪种规则的方法上面添加事务-->
            <tx:method name="accountMoney1" propagation="REQUIRED"/>
<!--            <tx:method name="account*"/>-->
        </tx:attributes>
    </tx:advice>
<!--    配置切入点和切面-->
    <aop:config>
<!--        配置切入点-->
        <aop:pointcut id="pt" expression="execution(* Spring_learning_package.TestDemo.TestShiWu.*(..))"/>
<!--        配置切面-->
        <aop:advisor advice-ref="txadvice" pointcut-ref="pt"/>
    </aop:config>
</beans>

- 完全注解开发(完全注解声明式事务管理)

@Configuration//配置类
@ComponentScan(basePackages = "Spring_learning_package")//开启组件扫描
@EnableTransactionManagement//开启事务
public class TxConfig {
    //创建数据库连接池
    @Bean
    public DruidDataSource getDruidDataSource(){
        DruidDataSource druidDataSource = new DruidDataSource();
        druidDataSource.setDriverClassName("com.mysql.jdbc.Driver");
        druidDataSource.setUrl("jdbc:mysql:///user_db?serverTimezone=UTC");
        druidDataSource.setUsername("root");
        druidDataSource.setPassword("hd200308041410");
        return druidDataSource;
    }
    //创建JdbcTemplate对象
    @Bean
    public JdbcTemplate getJdbcTemplate(DataSource dataSource){
        //在ioc容器中根据类型找到datasource
        JdbcTemplate jdbcTemplate = new JdbcTemplate();
        //注入datasource
        jdbcTemplate.setDataSource(dataSource);
        return jdbcTemplate;
    }
    //创建事务管理器
    @Bean
    public DataSourceTransactionManager getDataSourceTransactionManager(DataSource dataSource){
        DataSourceTransactionManager dataSourceTransactionManager = new DataSourceTransactionManager();
        dataSourceTransactionManager.setDataSource(dataSource);
        return dataSourceTransactionManager;
    }
}

此时在xml文件中的配置是:

<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.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 http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
    <!--    开启组件扫描-->
    <context:component-scan base-package="Spring_learning_package"/>
    <!-- 数据库连接池 -->
    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
          destroy-method="close">
        <property name="url" value="jdbc:mysql:///user_db?serverTimezone=UTC"/>
        <property name="username" value="root"/>
        <property name="password" value="hd200308041410"/>
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
    </bean>
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    <!--    创建事务管理器-->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <!--        注入数据源-->
        <property name="dataSource" ref="dataSource"/>
    </bean>
<!--    配置通知-->
    <tx:advice id = "txadvice">
        <tx:attributes>
<!--            指定在哪种规则的方法上面添加事务-->
            <tx:method name="accountMoney1" propagation="REQUIRED"/>
<!--            <tx:method name="account*"/>-->
        </tx:attributes>
    </tx:advice>
<!--    配置切入点和切面-->
    <aop:config>
<!--        配置切入点-->
        <aop:pointcut id="pt" expression="execution(* Spring_learning_package.TestDemo.TestShiWu.*(..))"/>
<!--        配置切面-->
        <aop:advisor advice-ref="txadvice" pointcut-ref="pt"/>
    </aop:config>
</beans>

- spring框架新功能

+ 日志功能

日志级别以及优先级排序: OFF > FATAL > ERROR > WARN > INFO > DEBUG > TRACE > ALL

  • 整个 Spring5 框架的代码基于 Java8 ,运行时兼容 JDK9,许多不建议使用的类和方
    法在代码库中删除

  • Spring 5.0 框架自带了通用的日志封装

    • Spring5 已经移除 Log4jConfigListener,官方建议使用 Log4j2

    • Spring5 框架整合 Log4j2

      • 第一步 引入 jar 包
        在这里插入图片描述

      • 第二步 创建 log4j2.xml 配置文件

<!-- 日志级别以及优先级排序: OFF > FATAL > ERROR > WARN > INFO > DEBUG > TRACE > ALL  -->
<!-- Configuration后面的status用于设置log4j2自身内部的信息输出,可以不设置,当设置成trace时,可以看到log4j2内部各种详细输出 -->
    <configuration status="INFO">
    <!-- 先定义所有的appender -->
    <appenders>
        <!-- 输出日志信息到控制台 -->
        <console name="Console" target="SYSTEM_OUT">
            <!-- 控制日志输出的格式 -->
            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n"/>
        </console>
    </appenders>
    <!-- 然后定义logger,只有定义了logger并引入的appender,appender才会生效 -->
    <!-- root:用于指定项目的根日志,如果没有单独指定Logger,则会使用root作为默认的日志输出 -->
    <loggers>
        <root level="info">
            <appender-ref ref="Console"/>
        </root>
    </loggers>
</configuration>

可以手动调节优先级以输出更多详细信息

日志信息如下:

2022-07-22 18:37:50.073 [main] INFO  Spring_learning_package.TestDemo.TestShiWu.UserLog - Hello
+ @Nullable 可以为空注解
  • @Nullable 注解可以使用在方法上面,属性上面,参数上面,表示方法返回可以为空,属性值可以为空,参数值可以为空

  • 注解用在方法上面,方法返回值可以为空

    在这里插入图片描述

  • 注解使用在方法参数里面,方法参数可以为空
    在这里插入图片描述

  • 注解使用在属性上面,属性值可以为空

在这里插入图片描述

+ 支持函数式风格GenericApplicationContex

在这里插入图片描述

+ 支持整合 JUnit5
  • 整合 JUnit4

    • 第一步 引入 Spring 相关针对测试依赖

在这里插入图片描述

  • 第二步 创建测试类,使用注解方式完成

在这里插入图片描述在这里插入图片描述

  • Spring5 整合 JUnit5

    • 第一步 引入 JUnit5 的 jar 包

在这里插入图片描述

  • 第二步 创建测试类,使用注解完成

在这里插入图片描述

  • 使用一个复合注解替代上面两个注解完成整合

在这里插入图片描述

+ Webflux介绍
  • SpringWebflux是 Spring5 添加新的模块,用于 web 开发的,功能和 SpringMVC 类似的,Webflux 使用当前一种比较流程响应式编程出现的框架。
    在这里插入图片描述

  • 使用传统 web 框架,比如 SpringMVC,这些基于 Servlet 容器,Webflux 是一种异步非阻
    塞的框架,异步非阻塞的框架在 Servlet3.1 以后才支持,核心是基于 Reactor 的相关 API 实现的。

` 什么是异步非阻塞
  • 异步和同步针对调用者,调用者发送请求,如果等着对方回应之后才去做其他事情就是同
    步,如果发送请求之后不等着对方回应就去做其他事情就是异步
  • 阻塞和非阻塞针对被调用者 阻塞和非阻塞针对被调用者,被调用者受到请求之后,做完请求任务之后才给出反馈就是阻塞,受到请求之后马上给出反馈然后再去做事情就是非阻塞
` Webflux 特点
  • 第一 非阻塞式:在有限资源下,提高系统吞吐量和伸缩性,以 Reactor 为基础实现响应式编程
  • 第二 函数式编程:Spring5 框架基于 java8,Webflux 使用 Java8 函数式编程方式实现路由请求
` 比较SpringMvc
  • 第一 两个框架都可以使用注解方式,都运行在 Tomet 等容器
  • 第二 SpringMVC 采用命令式编程,Webflux 采用异步响应式编程

在这里插入图片描述

- 响应式编程(Java实现)

  • 响应式编程是一种面向数据流和变化传播的编程范式。这意味着可以在编程语言中很方便地表达静态或动态的数据流,而相关的计算模型会自动将变化的值通过数据流进行传播。
    电子表格程序就是响应式编程的一个例子。单元格可以包含字面值或类似"=B1+C1"的公
    式,而包含公式的单元格的值会依据其他单元格的值的变化而变化。
+ java8及之前版本

提供的 观察者模式两个类 Observer 和 Observable
在这里插入图片描述

- 响应式编程(Reactor实现)

  • 响应式编程操作中,Reactor 是满足 Reactive 规范框架
  • Reactor 有两个核心类,Mono 和 和 Flux ,这两个类实现接口 Publisher,提供丰富操作
    符。Flux 对象实现发布者,返回 N 个元素;Mono 实现发布者,返回 0 或者 1 个元素
  • Flux 和 Mono 都是数据流的发布者,使用 Flux 和 Mono 都可以发出三种数据信号:
    元素值,错误信号,完成信号,错误信号和完成信号都代表终止信号,终止信号用于告诉
    订阅者数据流结束了,错误信号终止数据流同时把错误信息传递给订阅者

在这里插入图片描述

+ 代码演示 Flux 和 和 Mono

在这里插入图片描述

+ 三种信号特点
  • 错误信号和完成信号都是终止信号,不能共存的
  • 如果没有发送任何元素值,而是直接发送错误或者完成信号,表示是空数据流
  • 如果没有错误信号,没有完成信号,表示是无限数据流

注意:调用 just 或者其他方法只是声明数据流,数据流并没有发出,只有进行订阅之后才会触发数据流,不订阅什么都不会发生的

+ 操作符
  • 对数据流进行一道道操作,成为操作符,比如工厂流水线
  • 第一 map 元素映射为新元素

在这里插入图片描述

  • 第二 flatMap 元素映射为流
    把每个元素转换流,把转换之后多个流合并大的流

在这里插入图片描述

- SpringWebflux执行流程和核心API

SpringWebflux 基于 Reactor,默认使用容器是 Netty,Netty 是高性能的 NIO 框架,异步非阻
塞的框架

+ Netty
` BIO

在这里插入图片描述

` NIO

在这里插入图片描述

+ SpringWebflux 执行过程

SpringWebflux 执行过程和 SpringMVC 相似的

  • SpringWebflux 核心控制器 DispatchHandler,实现接口 WebHandler
  • 接口 WebHandler 有一个方法

在这里插入图片描述
在这里插入图片描述

+ SpringWebflux 中的 DispatcherHandler

SpringWebflux 里面 DispatcherHandler,负责请求的处理

  • HandlerMapping:请求查询到处理的方法
  • HandlerAdapter:真正负责请求处理
  • HandlerResultHandler:响应结果处理
+ SpringWebflux 函数式编程的两接口

SpringWebflux 实现函数式编程,两个接口:RouterFunction(路由处理)
和 HandlerFunction(处理函数)

- SpringWebflux(基于注解编程模型)

SpringWebflux 实现方式有两种:注解编程模型和函数式编程模型
使用注解编程模型方式,和之前 SpringMVC 使用相似的,只需要把相关依赖配置到项目中,
SpringBoot 自动配置相关运行容器,默认情况下使用 Netty 服务器

  • 第一步 创建 SpringBoot 工程,引入 Webflux 依赖

在这里插入图片描述

在这里插入图片描述

  • 第二步 配置启动端口号

  • 第三步 创建包和相关类

    实体类
    在这里插入图片描述

  • 创建接口定义操作的方法

在这里插入图片描述

  • 接口实现类

在这里插入图片描述
在这里插入图片描述

  • 创建 controller

在这里插入图片描述

  • 说明

    SpringMVC 方式实现,同步阻塞的方式,基于 SpringMVC+Servlet+Tomcat
    SpringWebflux 方式实现,异步非阻塞 方式,基于 SpringWebflux+Reactor+Netty

- SpringWebflux (基于函数式编程模型)

  • 在使用函数式编程模型操作时候,需要自己初始化服务器

  • 基于函数式编程模型时候,有两个核心接口:RouterFunction(实现路由功能,请求转发
    给对应的 handler)和 HandlerFunction(处理请求生成响应的函数)。核心任务定义两个函数
    式接口的实现并且启动需要的服务器。

  • SpringWebflux 请 求 和 响 应 不 再 是 ServletRequest 和 ServletResponse , 而 是
    ServerRequest 和 ServerResponse

  • 第一步 把注解编程模型工程复制一份 ,保留 entity 和 service 内容

  • 第二步 创建 Handler(具体实现方法)

  • 在这里插入图片描述
    在这里插入图片描述

  • 第三步 初始化服务器,编写 Router

    • 创建路由的方法

在这里插入图片描述

  • 创建服务器完成适配

在这里插入图片描述

  • 最终调用
    在这里插入图片描述

  • 第四步使用 WebClient 调用

在这里插入图片描述

总结


[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-GxeJLzoe-1659112249179)(C:\Users\lon\AppData\Roaming\Typora\typora-user-images\image-20220723191206214.png)]

rMapping:请求查询到处理的方法

  • HandlerAdapter:真正负责请求处理
  • HandlerResultHandler:响应结果处理
+ SpringWebflux 函数式编程的两接口

SpringWebflux 实现函数式编程,两个接口:RouterFunction(路由处理)
和 HandlerFunction(处理函数)

- SpringWebflux(基于注解编程模型)

SpringWebflux 实现方式有两种:注解编程模型和函数式编程模型
使用注解编程模型方式,和之前 SpringMVC 使用相似的,只需要把相关依赖配置到项目中,
SpringBoot 自动配置相关运行容器,默认情况下使用 Netty 服务器

  • 第一步 创建 SpringBoot 工程,引入 Webflux 依赖

在这里插入图片描述

在这里插入图片描述

  • 第二步 配置启动端口号

在这里插入图片描述

  • 第三步 创建包和相关类

    实体类

在这里插入图片描述

  • 创建接口定义操作的方法
    在这里插入图片描述

  • 接口实现类

在这里插入图片描述在这里插入图片描述

  • 创建 controller
    在这里插入图片描述

  • 说明

    SpringMVC 方式实现,同步阻塞的方式,基于 SpringMVC+Servlet+Tomcat
    SpringWebflux 方式实现,异步非阻塞 方式,基于 SpringWebflux+Reactor+Netty

- SpringWebflux (基于函数式编程模型)

  • 在使用函数式编程模型操作时候,需要自己初始化服务器

  • 基于函数式编程模型时候,有两个核心接口:RouterFunction(实现路由功能,请求转发
    给对应的 handler)和 HandlerFunction(处理请求生成响应的函数)。核心任务定义两个函数
    式接口的实现并且启动需要的服务器。

  • SpringWebflux 请 求 和 响 应 不 再 是 ServletRequest 和 ServletResponse , 而 是
    ServerRequest 和 ServerResponse

  • 第一步 把注解编程模型工程复制一份 ,保留 entity 和 service 内容

  • 第二步 创建 Handler(具体实现方法)
    在这里插入图片描述在这里插入图片描述

  • 第三步 初始化服务器,编写 Router

    • 创建路由的方法
      在这里插入图片描述

    • 创建服务器完成适配

      [外链图片转存中...(img-PhDh4GI3-1659112249177)]

    • 最终调用

  • 第四步使用 WebClient 调用

在这里插入图片描述

课程总结

在这里插入图片描述

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值