java实战,IOC,DI,过滤器,拦截器

准备工作

环境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相关注解总结

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值