Javaweb之Filter案例练习-项目全局编码过滤器

Filter全局编码过滤器

这篇来做一个Filter在全局编码过滤器练习,这篇新建一个web项目来做练习。

web项目创建

先创建一个叫GlobalFilter的web动态项目,把前面文章出现过的login.jsp复制过来,改成如下代码。

<%@ page language="java" contentType="text/html; charset=UTF-8"

   pageEncoding="UTF-8"%>

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "https://www.w3.org/TR/html4/loose.dtd">

<html>

<head>

<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">

<title>Insert title here</title>

</head>

<body>

    <form action="${pageContext.request.contextPath}/web/loginServlet" method="post">

        用户名:<input type="text" name="username" /><br/>

        <input type="submit" value="登录" /><br/>

    </form>

</body>

</html>

因为我们这里做全局编码的练习,到时候登录页面输入用户名,我们会采用中文字符输入。

Servlet创建

因为这里我们是模拟全局,实际中应该有许多servlet和许多前端页面,例如注册,登录,首页搜索等。我们就来创建一个servlet来模拟下就好。创建一个LoginServlet.java。 url-pattern是/web/loginServlet

package com.kaigejava.web.servlet;

import java.io.IOException;

import javax.servlet.ServletException;

import javax.servlet.annotation.WebServlet;

import javax.servlet.http.HttpServlet;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

@WebServlet("/loginServlet")

public class LoginServlet extends HttpServlet {    

    public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletExceptionIOException {

        //request.setCharacterEncoding("UTF-8");

        String name = request.getParameter("username");

        System.out.println(name);

    }    

    public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletExceptionIOException {

        doGet(request, response);

    }


}

以前,我们是通过这个红圈代码来设置编码,这样拿到中文的username就不会是乱码。

image1.png


现实开发中有很多个servlet,如果都这样写这行代码来设置编码,这行代码就冗余,我们可以把这行代码放到Filter过滤器中去。

Filter类创建

创建一个MyFilter.java 实现Filter接口,代码如下。

package com.kaigejava.web.filter;

import java.io.IOException;

import javax.servlet.Filter;

import javax.servlet.FilterChain;

import javax.servlet.FilterConfig;

import javax.servlet.ServletException;

import javax.servlet.ServletRequest;

import javax.servlet.ServletResponse;

import javax.servlet.annotation.WebFilter;

import javax.servlet.http.HttpServletRequest;

public class MyFilter implements Filter {

    public void destroy() {

        // TODO Auto-generated method stub

    }    

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

        // 转换对象

        HttpServletRequest req = (HttpServletRequest) request;

        // 设置编码

        req.setCharacterEncoding("UTF-8");

        // 放行

        chain.doFilter(request, response);

    }


    public void init(FilterConfig fConfig) throws ServletException {

        // TODO Auto-generated method stub

    }

}


web.xml内容如下

<?xml version="1.0" encoding="UTF-8"?>

<web-app xmlns:xsi="https://www.w3.org/2001/XMLSchema-instance" xmlns="https://java.sun.com/xml/ns/javaee" xsi:schemaLocation="https://java.sun.com/xml/ns/javaee https://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" id="WebApp_ID" version="3.0">

 <display-name>GlobalFilter</display-name>

 <welcome-file-list>

   <welcome-file>index.html</welcome-file>

   <welcome-file>index.htm</welcome-file>

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

   <welcome-file>default.html</welcome-file>

   <welcome-file>default.htm</welcome-file>

   <welcome-file>default.jsp</welcome-file>

 </welcome-file-list>

 

 <filter>

     <filter-name>MyFilter</filter-name>

     <filter-class>com.kaigejava.web.filter.MyFilter</filter-class>

 </filter>

 

 <filter-mapping>

     <filter-name>MyFilter</filter-name>

     <url-pattern>/*</url-pattern>

 </filter-mapping>

 

 <servlet>

   <servlet-name>LoginServlet</servlet-name>

   <servlet-class>com.anthony.web.servlet.LoginServlet</servlet-class>

 </servlet>

 <servlet-mapping>

   <servlet-name>LoginServlet</servlet-name>

   <url-pattern>/web/loginServlet</url-pattern>

 </servlet-mapping>

 

</web-app>


部署测试

现在部署到tomcat然后测试以下,前台页面用户名输入框输入“张三”看看。

image2.png

在Eclipse控制台可以看到“张三”打印出来不乱码,说明我们Filter中编码设置起效果了。

image3.png


Get方式提交表单问题

login.jsp中我们表单提交是post方法,如果改成get呢,会出问题。

image4.png

重新部署,再来测试下

image5.png

image6.png

如果是get方式传过来,中文就很有可能显示乱码。这个问题怎么解决了,有两种方式解决,第一种不要写get,就是post表单提交。现在我们来看看第二种方式如何解决。在MyFilter类中写一个通用的方法,既能解决get也能解决post的编码问题。


装饰模式解决get和post编码问题

我们在MyFilter.java代码中添加一个MyRequest class,里面写我们的装饰器代码,只重写了getParameter(String text)方法。

package com.kaigejava.web.filter;

import java.io.IOException;

import java.io.UnsupportedEncodingException;

import javax.servlet.Filter;

import javax.servlet.FilterChain;

import javax.servlet.FilterConfig;

import javax.servlet.ServletException;

import javax.servlet.ServletRequest;

import javax.servlet.ServletResponse;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletRequestWrapper;

public class MyFilter implements Filter {

    public void destroy() {

        // TODO Auto-generated method stub

    }    

    public void doFilter(ServletRequest requestServletResponse responseFilterChain chainthrows IOExceptionServletException {

        // 转换对象

        HttpServletRequest req = (HttpServletRequestrequest;

        // 设置编码

        //req.setCharacterEncoding("UTF-8");

        // 放行

        req = new MyRequest(req);

        // req这个对象是包装之后的对象,把这个传到servlet中的request

        chain.doFilter(reqresponse);

    }


    public void init(FilterConfig fConfigthrows ServletException {

        // TODO Auto-generated method stub

    }

}


/**

*  继承HttpServlet的包装类,实现装饰模式,解决getpost乱码问题

*

*/

class MyRequest extends HttpServletRequestWrapper{    

    HttpServletRequest request;    

    public MyRequest(HttpServletRequest request) {

        super(request);

        this.request = request;

    }    

    public String getParameter(String name) {

        name = request.getParameter(name); //乱码

        try {

            return new String(name.getBytes("iso-8859-1"), "UTF-8");

        catch (UnsupportedEncodingException e) {

            e.printStackTrace();

        }

        return null;

    }

    }



部署到tomcatlogin.jsp表单还是get提交,测试可以解决get和post乱码。

重写其他方法

在request对象中,还有其他方法,例如getParameters() getParameterMap(),这里我们把这两个方法都重写一下。

package com.kaigejava.web.filter;

import java.io.IOException;

import java.io.UnsupportedEncodingException;

import java.util.Map;

import javax.servlet.Filter;

import javax.servlet.FilterChain;

import javax.servlet.FilterConfig;

import javax.servlet.ServletException;

import javax.servlet.ServletRequest;

import javax.servlet.ServletResponse;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletRequestWrapper;

public class MyFilter implements Filter {

    public void destroy() {

    }

    public void doFilter(ServletRequest requestServletResponse responseFilterChain chainthrows IOExceptionServletException {

        // 转换对象

        HttpServletRequest req = (HttpServletRequestrequest;

        // 放行

        req = new MyRequest(req);

        // req变成自己包装的对象,然后传递给servlet中,servlet中的request就是调用包装过的req

        chain.doFilter(reqresponse);

    }

    public void init(FilterConfig fConfigthrows ServletException {

    }

}


/**

继承HttpServlet的包装类,实现装饰模式,解决getpost乱码问题

*

*/

class MyRequest extends HttpServletRequestWrapper {

        HttpServletRequest request;

        public MyRequest(HttpServletRequest request) {

        super(request);

        this.request = request;

    }

    

    /*@Override

    public String getParameter(String name) {

        name = request.getParameter(name); //乱码

        try {

            return new String(name.getBytes("iso-8859-1"), "UTF-8");

        } catch (UnsupportedEncodingException e) {

            e.printStackTrace();

        }

        return null;

    }*/

    

    @Override

    public String getParameter(String name) {

        Map<String, String[]> map = getParameterMap();

        return map.get(name)[0];

    }

    

    @Override

    public String[getParameterValues(String name) {

        Map<String, String[]> map = getParameterMap();

        return map.get(name);

    }

    

    public boolean flag = true;

    @Override

    public Map<String, String[]> getParameterMap() {


        Map<String, String[]> map = request.getParameterMap(); // 乱码

        if (flag) {


            for (Map.Entry<String, String[]> m : map.entrySet()) {

                String[values = m.getValue();

                for (int i = 0; i < values.lengthi++) {

                    try {

                        values[i] = new String(values[i].getBytes("iso-8859-1"), "UTF-8");

                    catch (UnsupportedEncodingException e) {

                        e.printStackTrace();

                    }

                }

            }

            flag = false;

        }

        return map;

    }

}


由于最后三个方法,我们倒数第二个第三个都是调用倒数第一个,为了解决第一个调用乱码好了,第二个再次调用又变成乱码,所以这里在第三个方法添加了一个flag



猜你喜欢

转载自blog.51cto.com/kaigejava/2430471