Spring5(五):Spring5新特性

1. 整合日志框架

  1. Spring5已经移除Log4jConfigListener,官方建议使用Log4j2
  2. Spring5框架整合Log4j2
    第一步 引入jar包
    在这里插入图片描述
    第二步 创建log4j2.xml配置文件
<?xml version="1.0" encoding="UTF-8"?> 
<!--日志级别以及优先级排序: 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>

2. @Nullable注解

  1. @Nullable注解可以使用在方法上面,属性上面,参数上面,表示方法返回可以为空,属性值可以为空,参数值可以为空
  2. 注解用在方法上面,方法返回值可以为空
    在这里插入图片描述
  3. 注解使用在方法参数里面,方法参数可以为空
    在这里插入图片描述
  4. 注解使用在属性上面,属性值可以为空
    在这里插入图片描述

3. 函数式注册对象

Spring5核心容器支持函数式风格GenericApplicationContext

//函数式风格创建对象,交给spring进行管理 
@Test 
public void testGenericApplicationContext() { 
	//1 创建GenericApplicationContext对象 
	GenericApplicationContext context = new GenericApplicationContext(); 
	//2 调用context的方法对象注册 
	context.refresh(); 
	context.registerBean("user1",User.class,() -> new User()); 
	//3 获取在spring注册的对象 
	// User user = (User)context.getBean("com.atguigu.spring5.test.User"); 
	User user = (User)context.getBean("user1"); 
	System.out.println(user); 
}

4. 整合JUnit5单元测试框架

4.1 整合JUnit4

  1. 引入Spring相关针对测试依赖
    在这里插入图片描述
    在这里插入图片描述
  2. 创建测试类,使用注解方式完成
@RunWith(SpringJUnit4ClassRunner.class) //单元测试框架 
@ContextConfiguration("classpath:bean1.xml") //加载配置文件
public class JTest4 { 
	@Autowired 
	private UserService userService; 
	@Test 
	public void test1() { 
		userService.accountMoney(); 
	} 
}

4.2 整合JUnit5

  1. 引入jar包
    在这里插入图片描述
  2. 创建测试类,使用注解完成
@ExtendWith(SpringExtension.class) 
@ContextConfiguration("classpath:bean1.xml") 
public class JTest5 { 
	@Autowired 
	private UserService userService; 
	@Test 
	public void test1() { 
		userService.accountMoney(); 
	} 
}

4.3 复合注释

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

@SpringJUnitConfig(locations = "classpath:bean1.xml") 
public class JTest5 { 
	@Autowired 
	private UserService userService; 
	@Test 
	public void test1() { 
		userService.accountMoney(); 
	} 
}

5. SpringWebflux使用

5.1 SpringWebflux介绍

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

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

  3. 解释什么是异步非阻塞
    异步和同步针对调用者,调用者发送请求,如果等着对方回应之后才去做其他事情就是同步,如果发送请求之后不等着对方回应就去做其他事情就是异步
    阻塞和非阻塞针对被调用者,被调用者受到请求之后,做完请求任务之后才给出反馈就是阻塞,受到请求之后马上给出反馈然后再去做事情就是非阻塞

  4. Webflux特点:
    第一 非阻塞式:在有限资源下,提高系统吞吐量和伸缩性,以Reactor为基础实现响应式编程
    第二 函数式编程:Spring5框架基于java8,Webflux使用Java8函数式编程方式实现路由请求

  5. 比较SpringMVC
    第一 两个框架都可以使用注解方式,都运行在Tomet等容器中
    第二 SpringMVC采用命令式编程,Webflux采用异步响应式编程
    在这里插入图片描述

5.2 响应式编程(Java实现)

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

  2. Java8及其之前版本
    提供的观察者模式两个类Observer和Observable

public class ObserverDemo extends Observable { 
	public static void main(String[] args) { 
		ObserverDemo observer = new ObserverDemo(); 
		//添加观察者 
		observer.addObserver((o,arg)->{ 
			System.out.println("发生变化"); 
		}); 
		observer.addObserver((o,arg)->{ 
			System.out.println("手动被观察者通知,准备改变"); 
		}); 
		observer.setChanged(); //数据变化 
		observer.notifyObservers(); //通知 
	} 
}

5.3 响应式编程(Reactor实现)

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

第一步 引入依赖

<dependency> 
	<groupId>io.projectreactor</groupId> 
	<artifactId>reactor-core</artifactId> 
	<version>3.1.5.RELEASE</version> 
</dependency>

第二步 编程代码

public static void main(String[] args) { 
	//just方法直接声明 
	Flux.just(1,2,3,4); 
	Mono.just(1); 

	//其他的方法 
	Integer[] array = {1,2,3,4}; 
	Flux.fromArray(array); 

	List<Integer> list = Arrays.asList(array); 
	Flux.fromIterable(list); 

	Stream<Integer> stream = list.stream(); 
	Flux.fromStream(stream); 
}
  1. 三种信号特点
    错误信号和完成信号都是终止信号,不能共存的
    如果没有发送任何元素值,而是直接发送错误或者完成信号,表示是空数据流
    如果没有错误信号,没有完成信号,表示是无限数据流
  2. 调用just或者其他方法只是声明数据流,数据流并没有发出,只有进行订阅之后才会触发数据流,不订阅什么都不会发生的
    在这里插入图片描述
  3. 操作符:对数据流进行一道道操作,成为操作符,比如工厂流水线
    第一步 map 元素映射为新元素
    在这里插入图片描述
    第二步 flatMap 元素映射为流。把每个元素转换流,把转换之后多个流合并大的流
    在这里插入图片描述

5.4 SpringWebflux执行流程和核心API

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

  1. Netty
    BIO
    在这里插入图片描述
    NIO
    在这里插入图片描述

  2. SpringWebflux执行过程和SpringMVC相似的
    SpringWebflux核心控制器 DispatchHandler,实现接口WebHandler
    接口WebHandler有一个方法
    在这里插入图片描述
    在这里插入图片描述

  3. SpringWebflux里面DispatcherHandler,负责请求的处理
    HandlerMapping:请求查询到处理的方法
    HandlerAdapter:真正负责请求处理
    HandlerResultHandler:响应结果处理

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

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

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

  1. 创建SpringBoot工程,引入Webflux依赖
    在这里插入图片描述
    在这里插入图片描述

  2. 配置启动端口号
    在这里插入图片描述

  3. 创建包和相关类
    实体类
    在这里插入图片描述
    创建接口定义操作的方法

//用户操作接口 
public interface UserService { 
	//根据id查询用户 
	Mono<User> getUserById(int id); 
	//查询所有用户 
	Flux<User> getAllUser(); 
	//添加用户 
	Mono<Void> saveUserInfo(Mono<User> user); 
}

接口实现类

public class UserServiceImpl implements UserService { 
	//创建map集合存储数据 
	private final Map<Integer,User> users = new HashMap<>(); 
	public UserServiceImpl() { 
		this.users.put(1,new User("lucy","nan",20)); 
		this.users.put(2,new User("mary","nv",30)); 
		this.users.put(3,new User("jack","nv",50)); 
	} 
	//根据id查询 
	@Override 
	public Mono<User> getUserById(int id) { 
		return Mono.justOrEmpty(this.users.get(id)); 
	} 
	//查询多个用户 
	@Override 
	public Flux<User> getAllUser() { 
		return Flux.fromIterable(this.users.values()); 
	}
	//添加用户 
	@Override public Mono<Void> saveUserInfo(Mono<User> userMono) { 
		return userMono.doOnNext(person -> { 
			//向map集合里面放值 
			int id = users.size()+1; 
			users.put(id,person); 
		}).thenEmpty(Mono.empty()); 
	} 
}

创建controller

@RestController 
public class UserController { 
	//注入service 
	@Autowired 
	private UserService userService; 
	//id查询 
	@GetMapping("/user/{id}") 
	public Mono<User> geetUserId(@PathVariable int id) { 
		return userService.getUserById(id); 
	} 
	//查询所有 
	@GetMapping("/user") 
	public Flux<User> getUsers() { 
		return userService.getAllUser(); 
	} 
	//添加 
	@PostMapping("/saveuser") 
	public Mono<Void> saveUser(@RequestBody User user) { 
		Mono<User> userMono = Mono.just(user); 
		return userService.saveUserInfo(userMono); 
	} 
}

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

(详看课件)
(1)在使用函数式编程模型操作时候,需要自己初始化服务器
(2)基于函数式编程模型时候,有两个核心接口:RouterFunction(实现路由功能,请求转发给对应的handler)和HandlerFunction(处理请求生成响应的函数)。核心任务定义两个函数式接口的实现并且启动需要的服务器。
(3)SpringWebflux请求和响应不再是ServletRequest和ServletResponse,而是ServerRequest和ServerResponse
第一步 把注解编程模型工程复制一份 ,保留entity和service内容
第二步 创建Handler(具体实现方法)

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

平什么阿

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值