Servlet与设计模式

1 过滤器和包装器

过滤器可以拦截请求及控制响应,而servlet对此毫无感知。过滤器有如下作用:

1)请求过滤器:完成安全检查、重新格式化请求首部或体、建立请求审计日志。

2)响应过滤器:压缩响应流、追加或修改响应流、创建一个完全不同的响应。

1.1 声明和确定过滤器顺序

当多个过滤器映射到一个给定资源时,容器会使用先URL模式后servlet-name的规则。

<filter-mapping>

<filter-name>filter3</filter-name>

<servlet-name>helloServlet</servlet-name>

</filter-mapping>

<filter-mapping>

<filter-name>filter1</filter-name>

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

</filter-mapping>

<filter-mapping>

<filter-name>filter2</filter-name>

<url-pattern>*.action</url-pattern>

</filter-mapping>

过滤器的顺序为: filter1、filter2、filter3

1.2 过滤器实战

需求:1)项目只接受/hello 请求的GET方法,对于其他请求则返回"404没有找到对应资源"。2)对所有响应,返回内容结尾都追加"filter实战"。

复制代码
public class HelloServlet extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        PrintWriter writer = resp.getWriter();
        writer.print("你好 Servlet");
        writer.close();
    }

}

public class RequestFilter implements Filter {

    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        System.out.println("拦截器初始化");
    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) servletRequest;
        String method = request.getMethod();
        String servletPath = request.getServletPath();
        HttpServletResponse response = (HttpServletResponse) servletResponse;
        CustomResponseWrapper responseWrapper = new CustomResponseWrapper(response);
        responseWrapper.setContentType("text/html;charset=UTF-8");
        if ("GET".equalsIgnoreCase(method) && "/hello".equalsIgnoreCase(servletPath)) {
            filterChain.doFilter(request,responseWrapper);
        } else {
            PrintWriter writer = responseWrapper.getWriter();
            writer.print("404没有找到对应资源");
            writer.close();
        }
    }

    @Override
    public void destroy() {
        System.out.println("拦截器销毁");
    }
}

public class CustomResponseWrapper extends HttpServletResponseWrapper {

    private CustomPrintWriter pw;

    public CustomResponseWrapper(HttpServletResponse response) {
        super(response);
    }

    @Override
    public PrintWriter getWriter() throws IOException {
        if (pw == null) pw = new CustomPrintWriter(super.getWriter());
        return pw;
    }
}

public class CustomPrintWriter extends PrintWriter {
    public CustomPrintWriter(Writer out) {
        super(out);
    }

    @Override
    public void print(String s) {
        super.print(s + "------filter实战");
    }
}

<web-app xmlns="http://java.sun.com/xml/ns/j2ee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"
         version="2.4">
    <servlet>
        <servlet-name>helloServlet</servlet-name>
        <servlet-class>HelloServlet</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>helloServlet</servlet-name>
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>

    <filter>
        <filter-name>reqFilter</filter-name>
        <filter-class>RequestFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>reqFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
</web-app>

web.xml

2 企业设计模式

Web系统要求着良好的性能,让用户得到更快的响应时间及支持更多的并发用户;为了让应用不同部分能同时在不同主机上运行,要以适当方式模块化;同时系统还要有一定的灵活性、可维护性和可扩展性。

2.1 处理远程对象

在对象跨网络通信时存在一些常见的问题:查找远程对象及处理本地和远程对象之间的底层网络I/O通信。也就是如何找到远程对象,以及如何调用其方法。

JNDI和RMI可以解决上述的问题。

2.1.1 JNDI

Java Naming and Directory Interface,Java命名和目录接口。可以在网络上的一个集中位置上完成查找。想被查找并访问的对象,要先向JNDI注册,再通过JNDI来查找。

2.1.2 为控制器增加JNDI。

图 WEB服务请求调用远程服务器服务过程

业务委托:完成请求的业务处理。

服务定位器:从JDNI查找需要调用的远程服务。

2.2 MVC

Model-View-Controller(模型-视图-控制器)。

模型:真正的业务逻辑和状态。了解获取和更新状态的规则。

视图:从控制器得到模型的状态。发起请求及接受响应。

控制器:从请求取得用户输入,并向模型明确输入用户的含义。

2.2.1 改进MVC控制器

控制器的3大任务及处理方案:

1)获得和处理请求参数,将这个任务交给另外一个单独的表单验证组件,由它获得表单参数、完成转换、进行验证、处理错误并创建一个对象保存参数值。

2)调用模型,采用声明方式,在定制部署描述文件中列出一些模型,控制器可以读取这些模型,并且根据请求来决定要使用哪些模型。

3)分派到视图,采用声明方式,根据URL控制器就能区分要分派到哪个视图。

前端控制器:应用所有的请求都会通过一个控制器,由它处理,并将请求分派到适当地方。

相关推荐
励志要当大牛的小白菜39 分钟前
ART配对软件使用
开发语言·c++·qt·算法
武子康1 小时前
Java-80 深入浅出 RPC Dubbo 动态服务降级:从雪崩防护到配置中心秒级生效
java·分布式·后端·spring·微服务·rpc·dubbo
爱装代码的小瓶子3 小时前
数据结构之队列(C语言)
c语言·开发语言·数据结构
YuTaoShao4 小时前
【LeetCode 热题 100】131. 分割回文串——回溯
java·算法·leetcode·深度优先
源码_V_saaskw4 小时前
JAVA图文短视频交友+自营商城系统源码支持小程序+Android+IOS+H5
java·微信小程序·小程序·uni-app·音视频·交友
Maybe_ch4 小时前
.NET-键控服务依赖注入
开发语言·c#·.net
超浪的晨4 小时前
Java UDP 通信详解:从基础到实战,彻底掌握无连接网络编程
java·开发语言·后端·学习·个人开发
终焉暴龙王4 小时前
CTFHub web进阶 php Bypass disable_function通关攻略
开发语言·安全·web安全·php
双力臂4045 小时前
Spring Boot 单元测试进阶:JUnit5 + Mock测试与切片测试实战及覆盖率报告生成
java·spring boot·后端·单元测试
Edingbrugh.南空5 小时前
Aerospike与Redis深度对比:从架构到性能的全方位解析
java·开发语言·spring