zoukankan      html  css  js  c++  java
  • servlet

    Servlet

    1. 什么是Servlet
      * Servlet是JavaWeb三大组件之一(Servlet、Filter、Listener)
      * Servlet是用来处理客户端请求的动态资源
      * Servlet的任务有:
        > 获取请求数据
        > 处理请求
        > 完成响应
      * Servlet接口方法:
        > void init(ServletConfig)
        > void service(ServletRequest,ServletResponse)
        > void destory()
        > ServletConfig getServletConfig()
        > String getServletInfo()

    2. 实现Servlet的方式
      * 实现Servlet接口(不方便)
      * 继承GenericServlet类(不方便)
      * 继承HttpServlet类(方便)

    3. Servlet第一例
      * 写一个类cn.example.MyServlet,实现Servlet接口
      * 实现service()方法,在其中给出System.out.println("hello servlet!");
      * 在web.xml文件中指定Servlet的访问路径为:/myservlet
    <servlet>
      <servlet-name>xxx</servlet-name>
      <servlet-class>com.example.MyServlet</servlet-class>
    </servlet>
    <servlet-mapping>
      <servlet-name>xxx</servlet-name>
      <url-pattern>/myservlet</url-pattern>
    </servlet-mapping>
      当用户在地址栏中访问:http://localhost:8080/example/myservlet时,会执行System.out.println("hello servlet!");

    ===============================

    Servlet生命周期

      Servlet接口一共5个方法,但其中只有三个是生命周期方法:
      * void init(ServletConfig)
      * void service(ServletRequest,ServletResponse)
      * void destory()

      1). 服务器创建Servlet:
      * 当Servlet第一次被请求时,或服务器启动时,服务器会创建Servlet实例。
      * 服务器默认是在servlet第一次被请求时创建Servlet实例,如果希望服务器启动时就创建Servlet实现需要在web.xml中配置
      * 服务器只为一个类型的Servlet创建一个实例对象,所以Servlet是单例的;

      2). 服务器初始化Servlet:
      * 当服务器创建Servlet实例后会马上调用Servlet的init(ServletConfig)方法,完成对Servlet的初始化;
      * init(ServletConfig)只会被调用一次
      * 服务器会在调用init()方法时传递ServletConfig参数
     
      3). 服务器使用Servlet处理请求:
      * 当Servlet被请求时,服务器会调用Servlet的service(ServletRequest,ServletResponse)方法
      * service(ServletRequest,ServletResponse)方法每处理一次请求,就会被调用一次,所以它可能会被调用N次
      * 因为Servlet是单例的,所以可能在同一时刻一个Servlet对象会被多个请求同时访问,所以这可能出现线程案例问题
      * Servlet不是线程安全的,这有助于提高效率,但不能让Servlet具有状态,以免多个线程争抢数据

      4). 服务器销毁Servlet
      * 服务器通常不会销毁Servlet,通常只有在服务器关闭时才会销毁Servlet
      * 服务器会在销毁Servlet之前调用Servlet的destory()方法
      * 可以在destory()方法中给出释放Servlet占有的资源,但通常Servlet是没什么可要释放的,所以该方法一般都是空的

    ===============================

    ServletConfig

      ServletConfig是Servlet中的init()方法的参数类型,服务器会在调用init()方法时传递ServletConfig对象给init()方法。
      ServletConfig对象封装了Servlet在web.xml中的配置信息,它对应<servlet>元素。ServletConfig类的功能有:
      * String getServletName():获取Servlet配置名,即<servlet-name>的值;
      * ServletContext getServletContext():获取ServletContext对象,这个对象稍后介绍
      * String getInitParameter(String name):获取初始化参数
      * Enumeration getInitParameterNames():获取所有初始化参数的名称

      在web.xml文件中,配置<servlet>时可以为<servlet>配置0~N个初始化参数,例如:
    <servlet>
      <servlet-name>xxx</servlet-name>
      <servlet-class>cn.example.servlet.MyServlet</servlet-class>
      <init-param>
        <param-name>p1</param-name>
        <param-value>v1</param-value>
      </init-param>
      <init-param>
        <param-name>p2</param-name>
        <param-value>v2</param-value>
      </init-param>
    </servlet>

    ===============================

    GenericServlet

      GenericServlet是Servlet接口的实现类,但它是一个抽象类,它唯一的抽象方法就是service()方法
      GenericServlet实现了Servlet方法:
      * 实现了String getServletInfo()方法
      * 实现了void destory()方法,空实现
      * 实现了void init(ServletConfig)方法,用来保存ServletConfig参数
      * 实现了ServletConfig getServletConfig()方法

      GenericServlet实现了ServletConfig接口:
      * 实现了ServletContext getServletContext()方法
      * 实现了String getInitParameter()方法
      * 实现了String getServletName()方法
      * 实现了Enumeration getInitParameterNames()方法

      GenericServlet添加了init()方法:
      * 该方法会被init(ServletConfig)方法调用
      * 如果希望对Servlet进行初始化,那么应该覆盖init()方法,而不是init(ServletConfig)方法

    ===============================

    HttpServlet
     
      HttpServlet是GenericServlet的子类,它专注HTTP请求
      HttpServlet类的方法:
      * 实现了void service(ServletRequest,ServletResponse)方法,实现内容是:
        > 把ServletRequest强转成HttpServletRequest
        > 把ServletResponse强转成HttpServletResponse
        > 调用本类添加的void service(HttpServletRequest,HttpServletResponse)方法
      * 添加了void service(HttpServletRequest,HttpServletResponse)方法,内容是:
        > 调用request的getMethod()获取请求方式
        > 如果请求方式为GET,那么调用本类添加的doGet(HttpServletRequest,HttpServletResponse)方法
        > 如果请求方式为POST,那么调用本类添加的doPost(HttpServletRequest,HttpServletResponse)方法
      * 添加了doGet(HttpServletRequest,HttpServletResponse)方法,内容是响应405,表示错误,所以我们应该去覆盖这个方法
      * 添加了doPost(HttpServletRequest,HttpServletResponse)方法,内容是响应405,表示错误,所以我们应用去覆盖这个方法

     如果是通过继承HttpServlet类来自定义Sevlet的话,那么:
      * 不要去覆盖void service(ServletRequest,ServletResponse)
      * 不要去覆盖void service(HttpServletRequest, HttpServletResponse)
      * 而应该去覆盖doGet()或doPost()方法。

    ===============================

    <url-pattern>

      <url-pattern>是<servlet-mapping>的子元素,用来绑定Servlet的访问路径
      可以在一个<servlet-mapping>中给出多个<url-pattern>,也就是说一个Servlet可以有多个访问路径:
      <servlet-mapping>
        <servlet-name>xxx</servlet-name>
        <url-pattern>/helo1<url-pattern>
        <url-pattern>/hello2<url-pattern>
      </servlet-mapping>

      还可以在<url-pattern>中使用通配符,即“*”。
      * <url-pattern>/*<url-pattern>:表示匹配任何路径
      * <url-pattern>/do/*<url-pattern>:表示匹配以/do开头的任何路径
      * <url-pattern>*.do<url-pattern>:表示匹配任何以“.do”结尾的路径

      注意:
      * 通配符要么在开头,要么在结尾,不能在中间,例如:/*.do就是错误的使用。
      * 如果不使用通配符,那么<url-pattern>必须以“/”开头,例如:<url-pattern>abc</url-pattern>就是错误的

    ===============================
    ===============================
    ===============================

    ServletContext
     
      ServletContext是Servlet三大域对象之一
      ServletContext在服务器启动时创建,在服务器关闭时销毁,一个JavaWeb应用只创建一个ServletContext对象
     
    1. 它的功能分类:
      * 存取数据
      * 读取web.xml中的应用初始化参数
      * 读取应用资源

    2. 获取ServletContext对象
      在HttpServlet中可以通过以下方法来获取ServletContext对象
      * ServletContext sc = this.getServletContext()
      * ServletContext sc = this.getServletConfig().getServletContext()

    2. 存取数据
      因为在一个JavaWeb应用中,只有一个ServletContext对象,所以在ServletContext中保存的数据可以共整个JavaWeb应用中的动态资源共享
      ServletContext是Servlet三大域对象之一,域对象内部有一个Map,用来保存数据

      * void setAttribute(String name, Object value):用来添加或替换ServletContext域数据
        > servletContext.setAttribute("xxx", "XXX"),添加域数据
        > servletContext.setAttribute("xxx", "XXXX"),覆盖域数据,因为在域中已经存在了名为xxx的数据,所以这次就是覆盖了
      * Object getAttribute(String name):通过名称来获取域数据
      * void removeAttribute(String name):通过名称移除域数据
      * Enumeration<String> getAttributeNames():获取所有ServletContext域数据的名称

    3. 读取web.xml中配置的应用初始化参数

      <context-param>
        <param-name>p1</param-name>
        <param-value>v1</param-value>      
      </context-param>
      <context-param>
        <param-name>p2</param-name>
        <param-value>v2</param-value>      
      </context-param>

      * servletContext.getInitParameter("p1"),返回v1
      * servletContext.getInitParameter("p2"),返回v2
      * servletContext.getInitParameterNames(),返回Enumeration<String>,包含p1和p2

    4. 获取项目资源
      * String getRealPath(String path):获取资源的真实名称
      String path = servletContext.getRealPath("/WEB-INF/a.jpg");
      返回值为/WEB-INF/a.jpg真实路径,即磁盘路径:C:/tomcat6/wabapps/hello/WEB-INF/a.jpg

      * InputStream getResourceAsStream(String path):获取资源的输入流
      InputStream in = servletContext.getResourceAsStream("/WEB-INF/a.jpg");
      返回的是a.jpg的输入流对象,可以从流中得到a.jpg的数据

      * Set<String> getResourcePaths(String path):获取指定目录下的所有资源路径
      Set<String> paths = servletContext.getResourcePaths("/WEB-INF");
      返回的Set中包含如下字符串:
        > /WEB-INF/lib/
        > /WEB-INF/classes/
        > /WEB-INF/web.xml
        > /WEB-INF/a.jpg

    ===============================
    ===============================
    ===============================

    获取类路径资源

      可以通过Class类的对象来获取类路径下的资源,对应JavaWeb应用的类路径就是classes目录下的资源
      例如:
      InputStream in = com.example.servlet.MyServlet.class.getResourceAsStream("a.jpg");
      获取的是:/WEB-INF/classes/com/example/servlet/a.jpg,即与MyServlet.class同目录下的资源

      例如:
      InputStream in = com.example.servlet.MyServlet.class.getResourceAsStream("/a.jpg");
      获取的是:/WEB-INF/classes/a.jpg,即类路径的根目录下的资源,类路径的根目录就是/classes目录

    不要在Servlet中创建成员!创建局部变量即可!
    可以创建无状态成员!
    可以创建有状态的成员,但状态必须为只读的!

     Servlet与线程安全
    因为一个类型的Servlet只有一个实例对象,那么就有可能会现时出一个Servlet同时处理多个请求,那么Servlet是否为线程安全的呢?答案是:“不是线程安全的”。这说明Servlet的工作效率很高,但也存在线程安全问题!
    所以我们不应该在Servlet中便宜创建成员变量,因为可能会存在一个线程对这个成员变量进行写操作,另一个线程对这个成员变量进行读操作。

    总结:
    1. Servlet生命周期方法是
      * init(ServletConfig):只被调用一次
      * service(ServletRequest,ServletResponse):被调用0~N次
      * destory():只被调用一次

    2. Http缺省请求方法是:GET

    3. HttpServlet对GET请求的处理方法是:doGet()

    4. <servlet>的子标签有:
      * <servlet-name>
      * <servlet-class>
      * <init-param>
      * <load-on-startup>

    5. ServletConfig的作用:
      * 对应<servlet>的配置信息
      * 可以获取<servlet>中配置的<init-param>信息

    6. Http响应头Content-type用来说明响应数据的MIME类型

    7. HttpServlet理解
      * HttpServlet是GenericServlet的子类,实现了GenericServlet中的抽象方法
      * service(ServletRequest, ServletResponse)会调用本类的service(HttpServletRequest,HttpServletResponse)
      * service(HttpServletRequest,HttpServletResponse)方法会根请求方式来调用相对的处理方法,例如请求方式为GET,那么该方法会调用doGet()方法,请求方式是POST,那么会调用doPost()方法。所以我们应该去重写HttpServlet的doGet()或doPost()方法。

    8. <url-pattern>通配符配置
      * <url-pattern>中可以使用通配符
      * 通配符要么在最前面,要么在最后,例如:*.do、/*都是正确的,但/*.jsp是错误的,因为*在中间,不是在最前面,也不是在最后面。
      * 必须以“/”或“*”开头

    9. ServletContext理解
      * 一个Web应用只有一个ServletContext对象,它会在服务器启动时创建,会在服务器关闭时被销毁,它的生命与服务器相同。
      * 通常在整个Web应用中共享数据时可以使用ServletContext对象
      * 还可以使用ServletContext来获取Web资源的真实路径,servletContext.getRealPath("/WEB-INF/a.jpg");
      * ServletContext可以用来资源的MIME类型,例如:servletContext.getMimeType("a.jpg"),它会返回image/jpeg

    10. <url-pattern>作用
      * 使用访问路径与Servlet绑定在一起

    11. 请求转发和重定向的异同
      * 请求转发是通过RequestDispatcher对象的forward()方法完成的
      * 重定向是通过HttpServletResponse对象的sendRediect()方法完成的
      * 请求转发是在一个请求中跨越多个动态资源(jsp/servlet),所以多个动态资源之间可以共享request数据
      * 重定向是两次请求,第一次请求服务器响应给客户端的是302,以及Location响应头,通知客户端再次去请求新的资源,所以客户端又发出第二次请求。所以重定向中被请求的多个动态资源之间不能共享request数据。
      * 请求转发后,地址栏的url不会改变,因为是一个请求;
      * 重定向后,地址的rul会改变,因为是两个请求

    12. 编写一个Servlet的可以通过哪些方式完成
      * 实现Servlet接口
      * 继承GenericServlet类
      * 继承HttpServlet类

    13. 在HttpServlet中获取ServletContext对象的方法有:
      * 通过ServletConfig类的getServletContext()方法;
      * HttpServlet类也提供了getServletContext()方法

    web.xml文件的继承(了解)
      在${CATALINA_HOME}confweb.xml中的内容,相当于写到了每个项目的web.xml中,它是所有web.xml的父文件。
    每个完整的JavaWeb应用中都需要有web.xml,但我们不知道所有的web.xml文件都有一个共同的父文件,它在Tomcat的conf/web.xml路径。
    conf/web.xml
    <?xml version="1.0" encoding="ISO-8859-1"?>

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

        <!-- default的优先级最低,如果一个请求没有人处理,那么它来处理!它显示404-->
            <servlet-name>default</servlet-name>

        <!-- 当访问路径不存在时,会执行DefaultServlet!访问index.html时也是在执行这个Servlet-->
            <servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-class>
            <init-param>
                <param-name>debug</param-name>
                <param-value>0</param-value>
            </init-param>
            <init-param>
                <param-name>listings</param-name>
                <param-value>false</param-value>
            </init-param>
            <load-on-startup>1</load-on-startup>
    </servlet>

        <servlet>
            <servlet-name>jsp</servlet-name>
            <servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class>
            <init-param>
                <param-name>fork</param-name>
                <param-value>false</param-value>
            </init-param>
            <init-param>
                <param-name>xpoweredBy</param-name>
                <param-value>false</param-value>
            </init-param>
            <load-on-startup>3</load-on-startup>
        </servlet>

        <servlet-mapping>
            <servlet-name>default</servlet-name>

        <!-- 匹配所有url,也就是说用户访问的url路径没有匹配的页面时,那么执行的就是名为default的Servlet,即org.apache.catalina.servlets.DefaultServlet-->
            <url-pattern>/</url-pattern>
        </servlet-mapping>

        <servlet-mapping>
            <servlet-name>jsp</servlet-name>

        <!-- 任何url后缀为jsp的访问,都会执行名为jsp的Servlet,即org.apache.jasper.servlet.JspServlet-->
            <url-pattern>*.jsp</url-pattern>
            <url-pattern>*.jspx</url-pattern>
        </servlet-mapping>

        <session-config>

    <!--session的默认超时时间为30分钟-->
            <session-timeout>30</session-timeout>
        </session-config>

        <!-- 这里省略了大概4000多行的MIME类型的定义,这里只给出两种MIME类型的定义,MIME用来标识网络上资源的媒体类型 -->
        <mime-mapping>
            <extension>bmp</extension>
            <mime-type>image/bmp</mime-type>
        </mime-mapping>
        <mime-mapping>
            <extension>htm</extension>
            <mime-type>text/html</mime-type>
        </mime-mapping>

        <welcome-file-list>

    <!-- 在应用的web.xml如果没有对<welcome-file-list>进行覆盖,那么默认主页为index.html,index.htm,index.jsp-->
            <welcome-file>index.html</welcome-file>
            <welcome-file>index.htm</welcome-file>
            <welcome-file>index.jsp</welcome-file>
        </welcome-file-list>
    </web-app>



  • 相关阅读:
    Linux性能监控
    程序员技术练级攻略
    使用 GDB 调试多进程程序
    nginx下面部署fast-cgi和C++【原】
    ROS Learning-024 (提高篇-002) rviz的安装和使用
    ROS Learning-023 (提高篇-001) 准备工作 --- 安装一些必要的软件包
    STM32 C++编程 005 I2c(Soft)类
    Python 网络爬虫 005 (编程) 如何编写一个可以 下载(或叫:爬取)一个网页 的网络爬虫
    设置 PyCharm 软件中 Terminal 窗口 中启动的 python 交互命令的版本
    在PyCharm 软件中设置你的项目 使用的Python版本
  • 原文地址:https://www.cnblogs.com/cyf18/p/14307412.html
Copyright © 2011-2022 走看看