1、日志
Spring 5.0框架自带了通用的日志封装,需要说明的是:
Spring5已经移除Log4jConfigListener,官方建议使用Log4j2
1、Spring5框架整合Log4j2
1、引入jar包
2、创建log4j2.xml配置文件(名字是固定的)
<?xml version="1.0" encoding="UTF-8"?>
<!--日志级别以及优先级排序: OFF > FATAL > ERROR > WARN > INFO > DEBUG > TRACE > ALL -->
<!--Configuration后面的status用于设置log4j2自身内部的信息输出,可以不设置,当设置成trace时,可以看到log4j2内部各种详细输出-->
<configuration status="ERROR">
<!--先定义所有的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、使用在方法上面
表示该方法的返回值可以为空
2、用在属性上面
表示属性值可以为空
3、用在方法参数上
表示参数可以为空
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.francis.spring5.pojo.User");
User user = (User)context.getBean("user1");
user.setName("张三");
System.out.println(user);
}
4、Spring5支持整合JUnit5
1、整个JUnit4
在原来的测试方式中,每次我们都要做重新去加载配置文件等重复的工作,整合JUnit4可以大大简化这个工过程
package com.francis.spring5.test;
import com.francis.spring5.service.UserService;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
/**
* @author Francis
* @create 2021-06-09-20:13
*/
@RunWith(SpringJUnit4ClassRunner.class) //单元测试框架
@ContextConfiguration("classpath:bean2.xml") //加载配置文件
public class Junit4Test {
@Autowired
private UserService userService;
@Test
public void transfer(){
userService.transfer(1, 2, 100);
}
}
同时需要这两个包
2、整个JUnit5
package com.francis.spring5.test;
import com.francis.spring5.service.UserService;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit.jupiter.SpringExtension;
import org.springframework.test.context.junit.jupiter.SpringJUnitConfig;
/**
* @author Francis
* @create 2021-06-09-20:40
*/
@ExtendWith(SpringExtension.class)
@ContextConfiguration("classpath:bean1.xml")
//复合注解,相当于上面两个,locations是一个数组,可以加载多个配置文件,更方便
//@SpringJUnitConfig(locations = "classpath:bean1.xml")
public class Junit5Test {
@Autowired
private UserService userService;
@Test
public void transfer(){
userService.transfer(1, 2, 100);
}
}
5、spring5新功能Webflux
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采用异步响应式编程
2、响应式编程(Java实现)
(1)什么是响应式编程响应式编程是一种面向数据流和变化传播的编程范式。这意味着可以在编程语言中很方便地表达静态或动态的数据流,而相关的计算模型会自动将变化的值通过数据流进行传播。电子表格程序就是响应式编程的一个例子。单元格可以包含字面值或类似"=B1+C1"的公式,而包含公式的单元格的值会依据其他单元格的值的变化而变化。
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(); //通知}}
}
}
这个新功能就不赘述了,需要时再学习。