Sevlet 中 Filter 的使用

网站开发过程中,一般都会采用前端发送网络请求,后端响应的模式,也即是 C/S 模式。通过 Filter 能够在 C/S 中建立一道闸门,拦截前端发送的请求。拦截后可以对请求进行预处理,然后在交个服务端进行处理。 Filter 在很多歌场景下都很有用。

Filter 生命周期

建立自己的 FilterMyFilter

public class MyFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        System.out.print("****** init ******");
    }

    //每次过滤都会执行的方法
    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        System.out.print("****** doFilter2 ******");
        filterChain.doFilter(servletRequest, servletResponse);
    }

    @Override
    public void destroy() {
        System.out.print("****** destroy ******");
    }
}

能够看到 MyFilter 中需要实现的 3 个方法 init、doFilter 和 destroy。 init 方法在 web 网站初始化的时候调用;doFilter在被过滤的对象执行时调用;destroy 在 MyFilter 销毁时调用。

Filter 在 web 网站的应用
  1. 通过 Filter 统一全站响应的数据编码格式
    web.xml文件中,将 MyFilter 设置为拦截全站所有的网络请求:
<filter>
    <filter-name>Filter1</filter-name>
    <filter-class>filter.MyFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>Filter1</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

MyFilterdoFilter 方法中设置数据编码格式:

public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
    HttpServletResponse res = (HttpServletResponse)servletResponse;
    servletResponse.setContentType("text/html;charset=" + encoding);
    filterChain.doFilter(servletRequest, servletResponse);      
}
  1. 控制浏览器缓存
    在 Filter 中针对图片设置一定时间的缓存时间,避免反复加载图片资源消耗带宽。
    web.xml 中设置 MyFilter 拦截图片资源的网络请求:
<filter>
    <filter-name>Filter1</filter-name>
    <filter-class>filter.MyFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>Filter1</filter-name>
    <url-pattern>*.png</url-pattern>
    <url-pattern>*.jpg</url-pattern>
    <url-pattern>*.gif</url-pattern>
    <url-pattern>*.bmp</url-pattern>
</filter-mapping>

MyFilterdoFilter 方法中设置数据编码格式:

public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
    HttpServletResponse res = (HttpServletResponse)servletResponse;
    HttpServletRequest req = (HttpServletRequest)servletRequest;
    //不缓存
    /*res.setHeader("cache-control", "no-cache");
    res.setHeader("pragma", "no-cache");
    res.setDateHeader("expires", 0);*/
    //缓存1小时
    res.setDateHeader("expires", System.currentTimeMillis() + 1000l*60*60);
    filterChain.doFilter(servletRequest, servletResponse);      
}
  1. 解决全站乱码问题
    get/post 请求中,参数默认采用ISO8859-1编码格式,会造成中文无法正常显示的错误。需要将请求参数改为utf-8格式,才能正常显示中文字符。
    使用 MyFilter 拦截所有网络请求,对 Request 采用包装类 MyHttpServletRequestWrapper 拦截调用的方法,重写 getParameterMap 方法。当在 request 中调用 getParameter 方法时,就会调用包装类中的方法,对中文进行自动转化。
public class MyFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest req = (HttpServletRequest)servletRequest;
        HttpServletResponse res = (HttpServletResponse)servletResponse;
        MyHttpServletRequestWrapper myRequest = new MyHttpServletRequestWrapper(req);
        filterChain.doFilter(myRequest,res);

    }

    @Override
    public void destroy() {

    }
}

class MyHttpServletRequestWrapper extends HttpServletRequestWrapper{

    private HttpServletRequest request;
    public MyHttpServletRequestWrapper(HttpServletRequest request) {
        super(request);
        this.request = request;
    }

    @Override
    public String getParameter(String name) {
        Map<String, String[]> map = getParameterMap();
        String[] values = map.get(name);
        if (values != null){
            return values[0];
        }else {
            return null;
        }
    }
    //Map 方法是基础方法
    private boolean hasEncoding = false;//防止反复解码;
    @Override
    public Map<String, String[]> getParameterMap() {
        String method = request.getMethod();
        if ("post".equalsIgnoreCase("method")){
            try{
                request.setCharacterEncoding("utf-8");
            }catch (UnsupportedEncodingException e){
                e.printStackTrace();
            }
        }else {
            Map<String, String[]> map = request.getParameterMap();
            Set<String> keys = map.keySet();

            if (!hasEncoding){
                hasEncoding = true;
                for (String key: keys){
                    String[] values = map.get(key);
                    for(int i = 0; i < values.length; i++){
                        try {
                            values[i] = new String(values[i].getBytes("ISO8859-1"),"UTF-8");

                        } catch (UnsupportedEncodingException e) {
                            e.printStackTrace();
                        }

                    }
                }
            }

        }

        return super.getParameterMap();

    }

    @Override
    public String[] getParameterValues(String name) {
        Map<String, String[]> map = getParameterMap();
        String[] values = map.get(name);
        return values;
    }
}
总结

Filter 相当于在前端的请求和后台服务间设立的一道大门,通过该大门可以实现请求的统一处理、个别请求和定制化处理和请求的放行和拒绝等功能。灵活使用 Filter 能够减少重复代码,加快网站开发效率。

关注和喜欢都是对我的鼓励和支持~
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 本文包括:1、Filter简介2、Filter是如何实现拦截的?3、Filter开发入门4、Filter的生命周期...
    廖少少阅读 7,352评论 3 56
  • IOC 控制反转容器控制程序对象之间的关系,而不是传统实现中,有程序代码之间控制,又名依赖注入。All 类的创建,...
    irckwk1阅读 987评论 0 0
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,973评论 19 139
  • Filter的基本功能是对Servlet容器调用Servlet的过程进行拦截,从而在Servlet进行响应处理的前...
    禅与发现的乐趣阅读 827评论 0 1
  • 手指同身寸,大概意思就是说本人的手指为标准来取穴.一种是中指同身寸,即中指中节内侧两端横纹间作为一寸,另一种为横指...
    月上梢头阅读 295评论 0 0