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}\conf\web.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>



posted on 2021-01-21 13:05  渐行渐远的那些人  阅读(159)  评论(0编辑  收藏  举报