美文网首页
210912:Bean初始化操作-SpringMVC中@Cont

210912:Bean初始化操作-SpringMVC中@Cont

作者: 弹钢琴的崽崽 | 来源:发表于2021-09-12 11:04 被阅读0次

    一. Bean初始化操作

    1. 简介

    很多时间当一个Bean被创建出来后,我们希望做一些初始化操作,如初始化数据、缓存预热等。有以下三种方法:

    • 初始化方法initMethod
    • 注解@PostConstruct
    • InitializingBeanafterPropertiesSet方法

    2. 三种方法实现

    先准备一个类用于测试,代码如下:

    public class BeanLifeCheck implements InitializingBean {
        private static final Logger logger = LoggerFactory.getLogger(BeanLifeCheck.class);
    
        @Value("${spring.application.name}")
        private String applicationName;
    
        public BeanLifeCheck() {
            logger.info("BeanLifeCheck: Construct " + applicationName);
        }
    
        public void initMethod() {
            logger.info("BeanLifeCheck: initMethod " + applicationName);
        }
    
        @PostConstruct
        public void postConstruct() {
            logger.info("BeanLifeCheck: postConstruct " + applicationName);
        }
    
        @Override
        public void afterPropertiesSet() throws Exception {
            logger.info("BeanLifeCheck: afterPropertiesSet " + applicationName);
        }
    }
    

    2.1 初始化方法initMethod

    这个以前是通过xml配置文件来定义的,现在可以直接定义在@Bean注解上,如下:

    @Bean(initMethod = "initMethod")
    public BeanLifeCheck beanLifeCheck() {
      return new BeanLifeCheck();
    }
    

    2.2 注解@PostConstruct

    直接在方法上加注解即可:

    @PostConstruct
    public void postConstruct() {
      logger.info("BeanLifeCheck: postConstruct " + applicationName);
    }
    

    2.3 InitializingBean的afterPropertiesSet方法

    需要类实现接口InitializingBean,如下:

    @Override
    public void afterPropertiesSet() throws Exception {
      logger.info("BeanLifeCheck: afterPropertiesSet " + applicationName);
    }
    

    3. 总结

    运行后的执行日志及顺序如下:

    c.r.springweb.day210911.BeanLifeCheck    : BeanLifeCheck: 构造方法 null
    c.r.springweb.day210911.BeanLifeCheck    : BeanLifeCheck: @postConstruct注解 testBeanInit
    c.r.springweb.day210911.BeanLifeCheck    : BeanLifeCheck: 实现InitializingBean接口afterPropertiesSet testBeanInit
    c.r.springweb.day210911.BeanLifeCheck    : BeanLifeCheck: initMethod testBeanInit
    

    二. SpringMVC中@ControllerAdvice注解的三种使用场景

    顾名思义,这是一个增强的 Controller。使用这个 Controller ,可以实现三个方面的功能:

    1. 全局异常处理
    2. 全局数据绑定
    3. 全局数据预处理

    灵活使用这三个功能,可以帮助我们简化很多工作,需要注意的是,这是 SpringMVC 提供的功能,在 Spring Boot 中可以直接使用,下面分别来看。

    1. 全局异常处理

    使用 @ControllerAdvice 实现全局异常处理,只需要定义类,添加该注解即可定义方式如下:

    @ControllerAdvice
    public class MyGlobalExceptionHandler {
        @ExceptionHandler(Exception.class)
        public String ExceptionInfo(Exception e){
            System.out.println("出现异常:"+e.getMessage());
            return "hello";
        }
    }
    

    1.1 Controller测试

    @RestController
    public class TestController {
        @GetMapping("/test")
        public String test(){
            int i = 1/0;
            System.out.println(i);
            return i+"";
        }
    }
    

    访问:http://127.0.0.1:8080/test

    1.2 总结

    在该类中,可以定义多个方法,不同的方法处理不同的异常,例如专门处理空指针的方法、专门处理数组越界的方法...,也可以直接向上面代码一样,在一个方法中处理所有的异常信息。

    @ExceptionHandler 注解用来指明异常的处理类型,即如果这里指定为 NullpointerException,则数组越界异常就不会进到这个方法中来。

    2. 全局数据绑定

    全局数据绑定功能可以用来做一些初始化的数据操作,我们可以将一些公共的数据定义在添加了 @ControllerAdvice 注解的类中,这样,在每一个 Controller 的接口中,就都能够访问导致这些数据。

    使用步骤,首先定义全局数据,如下:

    @ControllerAdvice
    public class MyGlobalExceptionHandler {
        @ModelAttribute(name = "md")
        public Map<String,Object> mydata() {
            HashMap<String, Object> map = new HashMap<>();
            map.put("age", 99);
            map.put("gender", "男");
            return map;
        }
    }
    

    使用 @ModelAttribute 注解标记该方法的返回数据是一个全局数据,默认情况下,这个全局数据的 key 就是返回的变量名,value 就是方法返回值,当然开发者可以通过 @ModelAttribute 注解的 name 属性去重新指定 key。

    定义完成后,在任何一个Controller 的接口中,都可以获取到这里定义的数据:

    @GetMapping("/hello")
    public String hello(Model model){
        Map<String, Object> map = model.asMap();
        System.out.println(map);
        System.out.println(map.get("md"));
        return "hello";
    }
    

    3. 全局数据预处理

    考虑我有两个实体类,Book 和 Author,分别定义如下:

    @Data
    public class Book {
        private String name;
        private Long price;
    }
    @Data
    public class Author {
        private String name;
        private Integer age;
    }
    

    这个时候,添加操作就会有问题,因为两个实体类都有一个 name 属性,从前端传递时 ,无法区分。此时,通过 @ControllerAdvice 的全局数据预处理可以解决这个问题

    解决步骤如下:

    3.1 给接口中的变量取别名

    @PostMapping("/book")
    public void addBook(@ModelAttribute("b") Book book, @ModelAttribute("a") Author author) {
        System.out.println(book);
        System.out.println(author);
    }
    

    3.2 进行请求数据预处理

    在 @ControllerAdvice 标记的类中添加如下代码:

    @InitBinder("b")
    public void b(WebDataBinder binder) {
        binder.setFieldDefaultPrefix("b.");
    }
    @InitBinder("a")
    public void a(WebDataBinder binder) {
        binder.setFieldDefaultPrefix("a.");
    }
    

    @InitBinder("b") 注解表示该方法用来处理和Book和相关的参数,在方法中,给参数添加一个 b 前缀,即请求参数要有b前缀.

    3.3 发送请求

    请求发送时,通过给不同对象的参数添加不同的前缀,可以实现参数的区分.

    打印如下

    三. lombok省略大量@Autowired

    在我们写controller或者Service层的时候,需要注入很多的mapper接口或者另外的service接口,这时候就会写很多的@AutoWired注解,代码看起来很乱
    lombok提供了一个注解:

    @RequiredArgsConstructor(onConstructor =@_(@Autowired))
    public class Test{
        // 写在类上可以代替@AutoWired注解,需要注意的是在注入时需要用final定义,或者使用@notnull注解
        private final User u;
    }
    

    四. SpringBoot之HandlerInterceptorAdapter

    在SpringBoot中我们可以使用HandlerInterceptorAdapter这个适配器来实现自己的拦截器。这样就可以拦截所有的请求并做相应的处理。

    1. 应用场景

    • 日志记录,可以记录请求信息的日志,以便进行信息监控、信息统计等。
    • 权限检查:如登陆检测,进入处理器检测是否登陆,如果没有直接返回到登陆页面。
    • 性能监控:典型的是慢日志。
    1. 在HandlerInterceptorAdapter中主要提供了以下的方法:
      preHandle:在方法被调用前执行。在该方法中可以做类似校验的功能。如果返回true,则继续调用下一个拦截器。如果返回false,则中断执行,也就是说我们想调用的方法 不会被执行,但是你可以修改response为你想要的响应。
    2. postHandle:在方法执行后调用。
    3. afterCompletion:在整个请求处理完毕后进行回调,也就是说视图渲染完毕或者调用方已经拿到响应。

    2. HandlerInterceptor

    2.1 拦截器适配器HandlerInterceptorAdapter

    public abstract class HandlerInterceptorAdapter implements AsyncHandlerInterceptor {
    
       /**
        * This implementation always returns {@code true}.
        */
       @Override
       public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
             throws Exception {
    
          return true;
       }
    
       /**
        * This implementation is empty.
        */
       @Override
       public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
             @Nullable ModelAndView modelAndView) throws Exception {
       }
    
       /**
        * This implementation is empty.
        */
       @Override
       public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler,
             @Nullable Exception ex) throws Exception {
       }
    
       /**
        * This implementation is empty.
        */
       @Override
       public void afterConcurrentHandlingStarted(HttpServletRequest request, HttpServletResponse response,
             Object handler) throws Exception {
       }
    
    }
    

    有时候我们可能只需要实现三个回调方法中的某一个,如果实现HandlerInterceptor接口的话,三个方法必须实现,不管你需不需要,此时spring提供了一个HandlerInterceptorAdapter适配器(种适配器设计模式的实现),允许我们只实现需要的回调方法。
    这样在我们业务中比如要记录系统日志,日志肯定是在afterCompletion之后记录的,否则中途失败了,也记录了,那就扯淡了。一定是程序正常跑完后,我们记录下那些对数据库做个增删改的操作日志进数据库。所以我们只需要继承HandlerInterceptorAdapter,并重写afterCompletion一个方法即可,因为preHandle默认是true。

    运行流程总结如下:

    1. 拦截器执行顺序是按照Spring配置文件中定义的顺序而定的。
    2. 会先按照顺序执行所有拦截器的preHandle方法,一直遇到return false为止,比如第二个preHandle方法是return false,则第三个以及以后所有拦截器都不会执行。若都是return true,则按顺序加载完preHandle方法。
    3. 然后执行主方法(自己的controller接口),若中间抛出异常,则跟return false效果一致,不会继续执行postHandle,只会倒序执行afterCompletion方法。
    4. 在主方法执行完业务逻辑(页面还未渲染数据)时,按倒序执行postHandle方法。若第三个拦截器的preHandle方法return false,则会执行第二个和第一个的postHandle方法和afterCompletion(postHandle都执行完才会执行这个,也就是页面渲染完数据后,执行after进行清理工作)方法。(postHandle和afterCompletion都是倒序执行)

    3. demo来演示执行流程

    3.1 定义一个类实现HandlerInterceptor,并重写方法

    快捷键ctrl+o打开可以重写的方法面板选择

    public class MyHandlerInterceptorAdapter implements HandlerInterceptor {
        @Override
        public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
            System.out.println("preHandle被执行");
            return true;
        }
    
        @Override
        public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
            System.out.println("psotHandle被执行");
        }
    
        @Override
        public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
            System.out.println("afterCompletion被执行");
    
        }
    }
    

    WebMvcConfigurerAdapter 抽象类是对WebMvcConfigurer接口的简单抽象(增加了一些默认实现),但在在SpringBoot2.0及Spring5.0中WebMvcConfigurerAdapter已被废弃 。官方推荐直接实现WebMvcConfigurer或者直接继承WebMvcConfigurationSupport

    3.2 实现WebMvcConfigurer配置拦截器

    @Configuration
    public class WebConfig implements WebMvcConfigurer {
        @Override
        public void addInterceptors(InterceptorRegistry registry) {
            registry.addInterceptor(new MyHandlerInterceptorAdapter());
        }
    }
    

    3.3 在控制器中写一个方法并访问

    @ApiOperation(value = "test mybatis", notes = "")
    @RequestMapping(value = "getuser", method = RequestMethod.GET)
    public User getUser() {
        return userService.getUser();
    }
    

    此时在加入一个拦截器,会按照配置的顺序执行,配置如下

    @Configuration
    public class WebConfig implements WebMvcConfigurer {
    
        @Override
        public void addInterceptors(InterceptorRegistry registry) {
            registry.addInterceptor(new MyHandlerInterceptorAdapter());
            registry.addInterceptor(new MyHandlerInterceptorAdapter2());
        }
    }
    

    控制的输出反映了配置多个拦截器的执行流程:

    如果controller出现异常,则不会继续执行postHandle,只会倒序执行afterCompletion方法

    相关文章

      网友评论

          本文标题:210912:Bean初始化操作-SpringMVC中@Cont

          本文链接:https://www.haomeiwen.com/subject/wlnzwltx.html