• spring boot中使用servlet、listener和filter


    spring boot中支持使用java Web三大组件(servlet、listener和filter),但是坑比较多,主要是spring boot内嵌tomcat和独立tomcat服务器有一些细节上的不同,踩完之后,特有此记。

    首先说明下,需要实现的功能,网站中有一些需要进行中英文对照的东西,一开始是写在properties配置文件中,经常修改的话,非常麻烦,还需要重启服务器。

    初步考虑觉得采用ServletConetext监听器的触发,从mysql表中加载所以中英文对照数据,存入map,保存在ServletContext中,这种方式简单有效,缺点在于如果一旦改动,就需要重启tomcat。

    进一步的想法是采用HttpSessionListener监听器,每个用户开始会话的session创建阶段,获取一次map,保存到ServletContext,所有的request都从全局变量ServletContext取,只存一份,实时更新,即使有变化也无需重启服务器。

    一、spring boot内嵌服务器

    1.session监听器的实现

    @WebListener
    public class MyHttpSessionListener implements HttpSessionListener {
        @Autowired
        private AppMetaService appMetaService;
    
        @Override
        public void sessionCreated(HttpSessionEvent httpSessionEvent) {
            Map<String, String> keyMap = appMetaService.getKeyMap();
            ServletContext servletContext = httpSessionEvent.getSession().getServletContext();
            servletContext.setAttribute("keyMap",keyMap);
        }
    
        @Override
        public void sessionDestroyed(HttpSessionEvent httpSessionEvent) {
        }
    }

    2.session监听无效

    这里有一个问题,发现controller中的请求并不会主动生成session,默认情况下session监听器什么也监听不到。所以需要在每次请求时,手动调用request.getSession触发session的创建。

    由于每次请求都需要这些动作,因此我把它加入到了一个过滤器中。

    @WebFilter(urlPatterns = "/*")
    public class MyFilter implements Filter {
        @Override
        public void init(FilterConfig filterConfig) throws ServletException {
        }
        /**
         * 在servlet过滤器中触发getSession方法,一次编写,多处使用
         */
        @Override
        public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
            HttpServletRequest request = (HttpServletRequest) servletRequest;
            request.getSession(true);
            filterChain.doFilter(request, servletResponse);
        }
    
        @Override
        public void destroy() {
    
        }
    }

    3.内嵌服务器支持

    Servlet、Filter、Listener可以直接通过@WebServlet、@WebFilter、@WebListener注解自动注册,无需其他代码。唯一需要的就是在项目的入口类中加一个注解@ServletComponentScan,顾名思义,它会扫描所有的
    @WebServlet、@WebFilter、@WebListener完成对象的注入。
    @SpringBootApplication
    @ServletComponentScan
    public class DataboardApplication {
       
    
        public static void main(String[] args) {
            SpringApplication.run(DataboardApplication.class, args);
        }
    }

    到此为止,简洁明了,不废话,本地测试成功,直接上tomcat服务器。

    二、独立tomcat服务器

    最大的坑来了,在本地跑的很正常的程序,在服务器上跑不动,最奇怪的是有些接口能用,有些不能用,经过多方观察,发现用到servlet、listener和filter的都挂了。

    搜了很多资料之后,还是在spring boot 2.01的文档中找到这么一句话:

    Tip
    @ServletComponentScan has no effect in a standalone container, where the container’s builtin
    discovery mechanisms are used instead.

    翻译一下就是@ServletComponentScan这个注解在独立tomcat服务器不可用。

    只好退求其次,采用@Bean注入的方式

    1.对应的监听器代码

    @Component
    public class MyHttpSessionListener implements HttpSessionListener {
        @Autowired
        private AppMetaService appMetaService;
    
        @Override
        public void sessionCreated(HttpSessionEvent httpSessionEvent) {
            Map<String, String> keyMap = appMetaService.getKeyMap();
            ServletContext servletContext = httpSessionEvent.getSession().getServletContext();
            servletContext.setAttribute("keyMap",keyMap);
        }
    
        @Override
        public void sessionDestroyed(HttpSessionEvent httpSessionEvent) {
        }
    }

    2.对应过滤器代码

    @Component
    public class MyFilter implements Filter {
        @Override
        public void init(FilterConfig filterConfig) throws ServletException {
        }
        /**
         * 在servlet过滤器中触发getSession方法,一次编写,多处使用
         */
        @Override
        public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
            HttpServletRequest request = (HttpServletRequest) servletRequest;
            request.getSession(true);
            filterChain.doFilter(request, servletResponse);
        }
    
        @Override
        public void destroy() {
    
        }
    }

    3.启动类代码

    @SpringBootApplication
    public class DataboardApplication {
        @Bean
        public ServletListenerRegistrationBean sessionHandler() {
            return new ServletListenerRegistrationBean<>(new MyHttpSessionListener());
        }
        
        @Bean
        public FilterRegistrationBean myFilter() {
            FilterRegistrationBean myFilter = new FilterRegistrationBean(new MyFilter());
            myFilter.addUrlPatterns("/*");
            return myFilter;
        }
    
        public static void main(String[] args) {
            SpringApplication.run(DataboardApplication.class, args);
        }
    }

    这里变化最大,去掉了@ServletComponentScan,在类内部增加了两个生成bean的方法。

    到此,大功告成!

  • 相关阅读:
    Add Binary
    Java笔记之String
    Java笔记之数组
    Merge Two Sorted Lists
    Remove Nth Node From End of List
    Longest Common Prefix
    Roman to Integer
    Palindrome Number
    Reverse Integer
    _cdel stdcall
  • 原文地址:https://www.cnblogs.com/wangbin2188/p/9199452.html
Copyright © 2020-2023  润新知