准备工作
环境SpringBoot、mybatis-plus
java实体类
@Data
public class User {
private Long id;
private String name;
private Integer age;
private String email;
}
controller
@RestController
@RequestMapping(value = "/user")
public class UserController {
@Autowired
private IUserService userService;
@RequestMapping(value = "test")
public String test(){
return "test";
}
@RequestMapping(value = "demo")
public String test01(){
return "demo";
}
}
application.yml文件自定义一个属性后面使用
myDemo:
testUrl: /test
spring/Springboot
IOC(向Spring容器中添加bean,默认单利)
@Component、@Service、@Controller、@Mapper常用在类上 ,经常使用不再多说
@Configuration与@Bean配合使用
bean的5种作用域
singleton、prototype、request、session、global session
@Configuration//说明改类是个配置文件
public class MyConfig {
@Bean//向Spring容器中放入User类型的bean
@RequestScope//(不写默认单利)指定bean的范围、请求域,@SessionScope:session域
public User user(){
return new User();
}
}
@Conditional系列注解,满足条件时才生效
与@Configuration搭配使用:满足条件配置才生效。
与@Bean搭配:满足条件才注册bean
@ConditionalOnBean(仅仅在当前上下文中存在某个对象时,才会实例化一个Bean)
@ConditionalOnClass(某个class位于类路径上,才会实例化一个Bean)
@ConditionalOnExpression(当表达式为true的时候,才会实例化一个Bean)
@ConditionalOnMissingBean(仅仅在当前上下文中不存在某个对象时,才会实例化一个Bean)
@ConditionalOnMissingClass(某个class类路径上不存在的时候,才会实例化一个Bean)
@ConditionalOnNotWebApplication(不是web应用,才会实例化一个Bean)
@ConditionalOnBean:当容器中有指定Bean的条件下进行实例化。
@ConditionalOnMissingBean:当容器里没有指定Bean的条件下进行实例化。
@ConditionalOnClass:当classpath类路径下有指定类的条件下进行实例化。
@ConditionalOnMissingClass:当类路径下没有指定类的条件下进行实例化。
@ConditionalOnWebApplication:当项目是一个Web项目时进行实例化。
@ConditionalOnNotWebApplication:当项目不是一个Web项目时进行实例化。
@ConditionalOnProperty:当指定的属性有指定的值时进行实例化。
@ConditionalOnExpression:基于SpEL表达式的条件判断。
@ConditionalOnJava:当JVM版本为指定的版本范围时触发实例化。
@ConditionalOnResource:当类路径下有指定的资源时触发实例化。
@ConditionalOnJndi:在JNDI存在的条件下触发实例化。
@ConditionalOnSingleCandidate:当指定的Bean在容器中只有一个,或者有多个但是指定了首选的Bean时触发实例化。
顶层接口
@FunctionalInterface
public interface Condition {
boolean matches(ConditionContext var1, AnnotatedTypeMetadata var2);
}
可自定义实现,注意导包
import org.springframework.context.annotation.Condition;
import org.springframework.context.annotation.ConditionContext;
import org.springframework.core.type.AnnotatedTypeMetadata;
public class TestCondition implements Condition
{
/**
* 只有返回true,才会启用配置
*/
public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata)
{
return true;
}
}
然后就可以使用了
@Configuration
@Conditional(TestCondition.class)
public class MyConfig {
@Bean
@Conditional(TestCondition.class)
public User user(){
return new User();
}
}
例子:根据配置文件中的值来判断是否注册bean
1.配置类如下
@Configuration
public class MyConfig {
@Bean
//1.name用来从yml或者(application.properties)中读取某个属性值。如果该值为空,则返回false
//2.如果值不为空,则将该值与havingValue指定的值进行比较,如果一样则返回true;否则返回false。
//3.matchIfMissing:如果是true就是第一步当没取到值也返回true
@ConditionalOnProperty(name = "myDemo.testUrl", havingValue = "/test", matchIfMissing = false)
public User user(){
return new User();
}
}
DI依赖注入(就是给bean(类)的属性赋值)
@Autowired
1.是Spring提供的,org.springframework.beans.factory.annotation,默认根据类型注入
2.默认情况下必须要求依赖对象必须存在,如果要允许null值,可以设置它的required属性为false,如@Autowired(required=false)
3.使用名称装配可以结合@Qualifier注解进行使用@Qualifier(“指定的实现类的bean名称”),指定使用那个实现类
实现原理:
@Autowired实现:
注解驱动配置会向spring容器中注册AutowiredAnnotationBeanPostProcessor
当 Spring 容器启动时,AutowiredAnnotationBeanPostProcessor 将扫描 Spring 容器中所有 Bean,当发现 Bean 中拥有 @Autowired 注释时就找到和其匹配(默认按类型匹配)的 Bean,并注入到对应的地方中去。
@Resource
1.JDK1.6支持的注解,默认按照名称进行装配,名称可以通过name属性进行指定。也提供按照byType 注入。
@Value注入yml(properties)文件中的值
@Value("${myDemo.testUrl}")
private String url;
过滤器:默认过滤所有请求 / *
1.定义类实现Filter
2.注册进Spring容器
3.指定要过滤的请求(可选,默认过滤所有请求)
4.@order注解不生效,需要使用FilterRegistrationBean指定
方式一 @Component注册(或者@Bean注册)用FilterRegistrationBean拓展
默认过滤所有请求
过滤指定请求,指定执行顺序:配合FilterRegistrationBean来扩展
注意不能与@WebFilter一起使用,否则会注册两个filter
@order注解不生效
import org.springframework.stereotype.Component;
import javax.servlet.*;
import java.io.IOException;
@Component
public class MyFilter implements Filter {
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
System.out.println("你的处理逻辑");
filterChain.doFilter(servletRequest, servletResponse);
}
}
配合FilterRegistrationBean扩展
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import java.util.Arrays;
@Configuration
public class MyConfig {
//注入@Component的bean,也可以使用@Bean注册
@Autowired
MyFilter myFilter;
//多个过滤器可以写多个方法,默认执行顺序从上到下
@Bean
public FilterRegistrationBean<MyFilter> filterRegistrationBean() {
FilterRegistrationBean<MyFilter> bean = new FilterRegistrationBean();
bean.setFilter(myFilter);
//指定过滤器的执行顺序,越小越先执行
bean.setOrder(1);
//要拦截的请求
String[] urls = {"/user/test/*", "/app"};
bean.setUrlPatterns(Arrays.asList(urls));
//给过滤器取个名字
bean.setName("myFilter");
return bean;
}
}
方式二@WebFilter配合@ServletComponentScan注册(无法指定过滤器执行顺序)
1.在定义的过滤器类上添加@WebFilter
2.启动类添加@ServletComponentScan
3.注意@Order注解无效
4.默认拦截所有路径,拦截指定路径使用@WebFilter的value属性
启动类添加@ServletComponentScan
@ServletComponentScan
@SpringBootApplication
public class AssemblyApplication {
public static void main(String[] args) {
SpringApplication.run(AssemblyApplication.class, args);
}
}
定义过滤器,并使用@WebFilte
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;
@WebFilter(value = {"/app", "/user/test/*"})
public class MyFilter implements Filter {
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
System.out.println("你的处理逻辑");
filterChain.doFilter(servletRequest, servletResponse);
}
}
拦截器(AOP,注意@Order无效,/*无效,需要使用/**)
1.定义类实现HandlerInterceptor,或者继承HandlerInterceptorAdapter
适配器模式(继承抽象类)好处:
实现接口需要重写接口所有的方法(default修饰的方法除外)
继承抽象类则不用2.所以选择继承HandlerInterceptorAdapter,实现我们需要的方法
3.将过滤器注册进入Spring容器
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@Component//注册进Spring容器
public class MyInterceptor extends HandlerInterceptorAdapter {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
throws Exception {
System.out.println("拦截器执行逻辑");
return true;
}
}
4.创建配置类并实现接口WebMvcConfigurer,注入拦截器,重写addInterceptors方法添加自定义拦截器
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
@Configuration
public class MyConfig implements WebMvcConfigurer {
@Autowired
MyInterceptor myInterceptor;
@Override
public void addInterceptors(InterceptorRegistry registry) {
//这里addInterceptor的顺序就是拦截器执行的顺序
//先执行
registry.addInterceptor(myInterceptor)
.addPathPatterns("/user/test/**", "/app")
.excludePathPatterns("/error");
//后执行
//registry.addInterceptor(myInterceptor1);
}
}
未完持续。。。。
SpringMVC
参数校验
@Validated
用在controller类上,配合jsr303注解使用
Lombok插件
指定使用类的某个属性EqualsAndHashCode方法的实现
@Data
@EqualsAndHashCode(of = {"id"})
public class User {
private Long id;
private String name;
private Integer age;
private String email;
}
隐试抛异常
@SneakyThrows
参考过的博客
@Autowired和@Resource的区别
@ConditionalOnProperty来控制Configuration是否生效
spring boot @ConditionalOnxxx相关注解总结