получение исключения IllegalStateException в методе doFilter

#java #spring #servlet-filters

#java #весна #сервлет-фильтры

Вопрос:

Когда я создаю фильтр сервлетов в Spring MVC, я получаю следующее исключение.

  [javax.servlet.ServletException: java.lang.IllegalStateException: Cannot create a session after the response has been committed] with root cause
  

когда я ввел системный вывод, я понял, что исключение возникает в строке перенаправления, но не понял, почему

может кто-нибудь, пожалуйста, подсказать мне какое-нибудь решение для этого

SessionFilter.java

 public class SessionFilter implements Filter {

    private ArrayList<String> urlList;

    @Override
    public void destroy() {
    }

    @Override
    public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) req;
        HttpServletResponse response = (HttpServletResponse) res;
        String url = request.getServletPath();
        boolean allowedRequest = false;

        if(urlList.contains(url)) {
            allowedRequest = true;
        }

        if (!allowedRequest) {
            HttpSession session = request.getSession(false);
            if (null == session) {
                System.out.println("preparing for redirect");
                response.sendRedirect("index.jsp");
            }
        }

        chain.doFilter(req, res);
    }

    @Override
    public void init(FilterConfig config) throws ServletException {
        System.out.println("entered init");
        String urls = config.getInitParameter("avoid-urls");
        StringTokenizer token = new StringTokenizer(urls, ",");

        urlList = new ArrayList<String>();

        while (token.hasMoreTokens()) {
            urlList.add(token.nextToken());

        }
    }
  

web.xml

 <?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
  <display-name>SpringMVCHibernate</display-name>

  <welcome-file-list>
    <welcome-file>index.jsp</welcome-file>
  </welcome-file-list>

  <servlet>
        <servlet-name>dispatcher</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>dispatcher</servlet-name>
        <url-pattern>*.do</url-pattern>
    </servlet-mapping>

    <session-config>
        <session-timeout>1</session-timeout>
    </session-config>


    <filter>
        <filter-name>SessionFilter</filter-name>
        <filter-class>com.common.dao.SessionFilter</filter-class>
        <init-param>
            <param-name>avoid-urls</param-name>
            <param-value>index.jsp</param-value>
        </init-param>
    </filter>

    <filter-mapping>
        <filter-name>SessionFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
:
:
:
  

Комментарии:

1. Опубликуйте полную трассировку стека.

Ответ №1:

К простому и короткому ответу на ваш вопрос:

Чтобы избежать этого, у вас должен быть оператор return , или избегайте перенаправления или пересылки запроса, или подобные вещи должны выполняться последним фильтром в цепочке фильтров.

Более подробное объяснение вы можете прочитать ниже :

Это исключение возникает при повторной попытке отправить ответ, когда ответ уже зафиксирован и отправлен пользователю. Например :

Здесь, в приведенном ниже примере кода, сначала запрос будет перенаправлен в index_test.jsp, и ответ будет отправлен пользователю, затем элемент управления снова вернется к filter и попытается отправить другой ответ (перенаправление) пользователю, и он завершится неудачей.

Обычно мы проверяем несколько условий в filter и, соответственно, перенаправляем и перенаправляем, если выполняются два условия, это создаст проблему .

 public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { 

  request.getRequestDispatcher("index_test.jsp").forward(request, response);
  ((HttpServletResponse)response).sendRedirect("new.jsp");
  chain.doFilter(request, response);
 }
  

Решение для вашего случая :

Чтобы избежать этого, у вас должен быть оператор return , или избегайте перенаправления или пересылки запроса, или подобные вещи должны выполняться последним фильтром в цепочке фильтров.

Таким образом, вы можете изменить свой код следующим образом :

 public class SessionFilter implements Filter {

    private ArrayList<String> urlList;

    @Override
    public void destroy() {
    }

    @Override
    public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) req;
        HttpServletResponse response = (HttpServletResponse) res;
        String url = request.getServletPath();
        boolean allowedRequest = false;

        if(urlList.contains(url)) {
            allowedRequest = true;
        }

        if (!allowedRequest) {
            HttpSession session = request.getSession(false);
            if (null == session) {
                System.out.println("preparing for redirect");
                response.sendRedirect("index.jsp");
                return;
            }
        }

        chain.doFilter(req, res);
    }

    @Override
    public void init(FilterConfig config) throws ServletException {
        System.out.println("entered init");
        String urls = config.getInitParameter("avoid-urls");
        StringTokenizer token = new StringTokenizer(urls, ",");

        urlList = new ArrayList<String>();

        while (token.hasMoreTokens()) {
            urlList.add(token.nextToken());

        }
    }
  

Комментарии:

1. Надеюсь, вы понимаете мой пример.

2. С какой проблемой вы столкнулись сейчас? Опубликовать полную трассировку стека

3. просто получение preparing for redirect печати

Ответ №2:

Его очевидное право :

 System.out.println("preparing for redirect");
   response.sendRedirect("index.jsp");
   return; 
  

Мы возвращаемся после этого утверждения.
Таким образом, дальнейшее выполнение не произойдет, поэтому вы не получаете исключение.

Смотрите, мы не можем использовать как фильтрацию, так и перенаправление или пересылку в одном блоке, это приведет к исключению IllegalStateException, потому что: Это исключение возникает при попытке повторно отправить ответ, когда ответ уже зафиксирован и отправлен пользователю