移动软件开发技术博客(Andoroid & ISO)

计划比目标更重要,行动比决心更重要

Servlet简介 以及servlet一些细节 servletconfig servletcontext 的几个应用

 

Tip:Servlet简介

Servlet是sun公司提供的一门用于开发动态web资源的技术。

Sun公司在其API中提供了一个servlet接口,用户若想用发一个动态web资源(即开发一个Java程序向浏览器输出数据),需要完成以下2个步骤:

编写一个Java类,实现servlet接口。

把开发好的Java类部署到web服务器中。

快速入门,用servlet向浏览器输出“hello servlet”。

阅读Servlet API,解决两个问题:

输出hello servlet的java代码应该写在servlet的哪个方法内?

如何向IE浏览器输出数据?

新建一个web应用,在web应用的WEB-INF\classes目录中,写一个java实现servlet接口,并覆盖service方法

package cn.itcast;

import java.io.*;

import javax.servlet.*;

public class ServletDemo1 extends GenericServlet{

public void service(ServletRequest req,ServletResponse res) throws ServletException,java.io.IOException

       {

              OutputStream out = res.getOutputStream();

              out.write("hello servlet!!!".getBytes());

       }

}

2.把servlet相关的jar加入构建路径  javac只是把J2SE的包API导入

set classpath=%classpath%;C:\apache-tomcat-6.0.20\lib\servlet-api.jar;

3.进入到ServletDemo1.java所在的目录,编译ServletDemo1.java

javac -d . ServletDemo1.java

4.在web工程下的WEB-INF目录中新建web.xml文件,配置servlet的对外访问路径

<?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_2_5.xsd"

    version="2.5">

      

       <servlet>

               <servlet-name>ServletDemo1</servlet-name>

               <servlet-class>cn.itcast.ServletDemo1</servlet-class>

           </servlet>

       <servlet-mapping>

               <servlet-name>ServletDemo1</servlet-name>

               <url-pattern>/ServletDemo1</url-pattern>

           </servlet-mapping>

   

</web-app>

5.启动tomcat和ie访问: http://localhost:8080/day05/ ServletDemo1

Servlet 调用图

Servlet的运行过程

Servlet程序是由WEB服务器调用,web服务器收到客户端的Servlet访问请求后:

1 Web服务器首先检查是否已经装载并创建了该Servlet的实例对象。如果是,则直接执行第4步,否则,执行第2步。

2装载并创建该Servlet的一个实例对象。

3调用Servlet实例对象的init()方法。

4创建一个用于封装HTTP请求消息的HttpServletRequest对象和一个代表HTTP响应消息的HttpServletResponse对象,然后调用Servlet的service()方法并将请求和响应对象作为参数传递进去。

WEB应用程序被停止或重新启动之前,Servlet引擎将卸载Servlet,并在卸载之前调用Servlet的destroy()方法。

ServletDemo1 extends GenericServlet  注意导入源码 是tomacat的src文件

Servlet接口实现类

Servlet接口SUN公司定义了两个默认实现类,分别为:GenericServlet、HttpServlet。

HttpServlet指能够处理HTTP请求的servlet,它在原有Servlet接口上添加了一些与HTTP协议处理方法,它比Servlet接口的功能更为强大。因此开发人员在编写Servlet时,通常应继承这个类,而避免直接去实现Servlet接口。

HttpServlet在实现Servlet接口时,覆写了service方法,该方法体内的代码会自动判断用户的请求方式,如为GET请求,则调用HttpServlet的doGet方法,如为Post请求,则调用doPost方法。因此,开发人员在编写Servlet时,通常只需要覆写doGet或doPost方法,而不要去覆写service方法。

某个对象在生命周期内必定要执行的方法 叫做与生命周期相关的方法

生命周期: 什么时候生 什么时候死 生命周期中执行的方法

有些数据不适合在程序中写出 可以通过配置文件写

servletConfig对象:用于封装servlet的配置信息

在实际开发,有一些东西不适合在servlet程序中写死,这类数据就可以通过配置方式配给servlet,例如servlet采用哪个码表、servlet连接哪个库、servlet哪个配置文件

servlet的生命周期

服务器第一次被servlet请求时 servlet会生 然后便在内存 响应 再来请求的时候不需要再创建对象 第一次创建的会一直留在内存中响应 请求 init方法 service方法进行服务 被摧毁时destory方法关掉服务器 web应用被删除的时候 会死 10:00

改名字的时候要用 refactor 改名字 不过还得把web.xml再改下

改模板 到eclipse文件夹 然后上一步 搜索servlet.java 然后更改

小心名字问题

Servlet的一些细节(1)

由于客户端是通过URL地址访问web服务器中的资源,所以Servlet程序若想被外界访问,必须把servlet程序映射到一个URL地址上,这个工作在web.xml文件中使用<servlet>元素和<servlet-mapping>元素完成。

<servlet>元素用于注册Servlet,它包含有两个主要的子元素:<servlet-name>和<servlet-class>,分别用于设置Servlet的注册名称和Servlet的完整类名。

一个<servlet-mapping>元素用于映射一个已注册的Servlet的一个对外访问路径,它包含有两个子元素:<servlet-name>和<url-pattern>,分别用于指定Servlet的注册名称和Servlet的对外访问路径。例如:

Servlet的一些细节(2)

同一个Servlet可以被映射到多个URL上,即多个<servlet-mapping>元素的<servlet-name>子元素的设置值可以是同一个Servlet的注册名。

在Servlet映射到的URL中也可以使用*通配符,但是只能有两种固定的格式:一种格式是“*.扩展名”,另一种格式是以正斜杠(/)开头并以“/*”结尾。

Exp:*.do   /action/*

Servlet的一些细节(3)

对于如下的一些映射关系:

Servlet1 映射到 /abc/*

Servlet2 映射到 /*

Servlet3 映射到 /abc

Servlet4 映射到 *.do

问题:

当请求URL为“/abc/a.html”,“/abc/*”和“/*”都匹配,哪个servlet响应

       Servlet引擎将调用Servlet1。

当请求URL为“/abc”时,“/abc/*”和“/abc”都匹配,哪个servlet响应

       Servlet引擎将调用Servlet3。

当请求URL为“/abc/a.do”时,“/abc/*”和“*.do”都匹配,哪个servlet响应

       Servlet引擎将调用Servlet1。

当请求URL为“/a.do”时,“/*”和“*.do”都匹配,哪个servlet响应

       Servlet引擎将调用Servlet2。

当请求URL为“/xxx/yyy/a.do”时,“/*”和“*.do”都匹配,哪个servlet响应

      

Servlet引擎将调用Servlet2。

*在前优先度最低

伪静态: 动态web资源映射到静态的web资源上

Servlet是一个供其他Java程序(Servlet引擎)调用的Java类,它不能独立运行,它的运行完全由Servlet引擎来控制和调度。

针对客户端的多次Servlet请求,通常情况下,服务器只会创建一个Servlet实例对象,也就是说Servlet实例对象一旦创建,它就会驻留在内存中,为后续的其它请求服务,直至web容器退出,servlet实例对象才会销毁。

如果在<servlet>元素中配置了一个<load-on-startup>元素,那么WEB应用程序在启动时,就会装载并创建Servlet的实例对象、以及调用Servlet实例对象的init()方法。

       举例:

       <servlet>

              <servlet-name>invoker</servlet-name>

              <servlet-class>

                     org.apache.catalina.servlets.InvokerServlet

              </servlet-class>

              <load-on-startup>2</load-on-startup>

       </servlet>

用途:为web应用写一个InitServlet,这个servlet配置为启动时装载,为整个web应用创建必要的数据库表和数据。服务器启动时初始化servlet 为servlet配置<load-on-startup>   数字越小越先启动 优先级越高

如果某个Servlet的映射路径仅仅为一个正斜杠(/),那么这个Servlet就成为当前Web应用程序的缺省Servlet。url-pattern 后面不写地址 就是缺省的地址

凡是在web.xml文件中找不到匹配的<servlet-mapping>元素的URL,它们的访问请求都将交给缺省Servlet处理,也就是说,缺省Servlet用于处理所有其他Servlet都不处理的访问请求。

在<tomcat的安装目录>\conf\web.xml文件中,注册了一个名称为org.apache.catalina.servlets.DefaultServlet的Servlet,并将这个Servlet设置为了缺省Servlet。

当访问Tomcat服务器中的某个静态HTML文件和图片时,实际上是在访问这个缺省Servlet

自己就千万别设置缺省的了 那样外部访问web资源就不能看到静态html文件以及图片了

Servlet的一些细节(7)—线程安全

当多个客户端并发访问同一个Servlet时,web服务器会为每一个客户端的访问请求创建一个线程,并在这个线程上调用Servlet的service方法,因此service方法内如果访问了同一个资源的话,就有可能引发线程安全问题。

如果某个Servlet实现了SingleThreadModel接口,那么Servlet引擎将以单线程模式来调用其service方法。

SingleThreadModel接口中没有定义任何方法(标记接口 : 只有个接口没有方法啥都没定义 实现这样的接口就好像类打了一个标号 就有一些功能了),只要在Servlet类的定义中增加实现SingleThreadModel接口的声明即可。 

对于实现了SingleThreadModel接口的Servlet,Servlet引擎仍然支持对该Servlet的多线程并发访问,其采用的方式是产生多个Servlet实例对象,并发的每个线程分别调用一个独立的Servlet实例对象。

实现SingleThreadModel接口并不能真正解决Servlet的线程安全问题,因为Servlet引擎会创建多个Servlet实例对象,而真正意义上解决多线程安全问题是指一个Servlet实例对象被多个线程同时调用的问题。事实上,在Servlet API 2.4中,已经将SingleThreadModel标记为Deprecated(过时的)。  static 有线程安全 还有内存崩溃的情况发生

静态的东西一定要小心 有可能有内存爆的问题

Javaweb传参数的方式

  1. 传统方式 response request
  2. 把参数放在ThreadLocal这个容器里 线程走到哪里就可以在哪里取出参数
  3. 在java虚拟机启动时 启动jndi容器 然后参数与这个容器进行绑定 谁要取 谁叫到这个容器里拿

线程安全问题发生在多个资源操作共享数据的时候

web的异常要抓起来不能往外扔

子类在覆盖父类的方法不能比父类抛出更多的异常 但是可以catch住

在Servlet的整个生命周期内,Servlet的init方法只被调用一次(应为对象只是创建一个)。而对一个Servlet的每次访问请求都导致Servlet引擎调用一次servlet的service方法。对于每次访问请求,Servlet引擎都会创建一个新的HttpServletRequest请求对象和一个新的HttpServletResponse响应对象,然后将这两个对象作为参数传递给它调用的Servlet的service()方法,service方法再根据请求方式分别调用doXXX方法。

ServletConfig对象

在Servlet的配置文件中,可以使用一个或多个<init-param>标签为servlet配置一些初始化参数。context - param 为整个web应用配置初始化参数

l  当servlet配置了初始化参数后,web容器在创建servlet实例对象时,会自动将这些初始化参数封装到ServletConfig对象中,并在调用servlet的init方法时,将ServletConfig对象传递给servlet。进而,程序员通过ServletConfig对象就可以得到当前servlet的初始化参数信息。

servletConfig对象:用于封装servlet的配置信息

//在实际开发,有一些东西不适合在servlet程序中写死,这类数据就可以通过配置方式配给servlet,例如

//  servlet采用哪个码表、servlet连接哪个库、servlet哪个配置文件

ServletContext

WEB容器在启动时,它会为每个WEB应用程序都创建一个对应的ServletContext对象,它代表当前web应用。

l  ServletConfig对象中维护了ServletContext对象的引用,开发人员在编写servlet时,可以通过ServletConfig.getServletContext方法获得ServletContext对象。

l  由于一个WEB应用中的所有Servlet共享同一个ServletContext对象,因此Servlet对象之间可以通过ServletContext对象来实现通讯。ServletContext对象通常也被称之为context域对象

l  查看ServletContext API文档,了解ServletContext对象的功能

servletContext实现的是请求转发 数据不能让它带 因为它都给共享了 要通过request域

域:容器 应用程序的范围

实现数据共享 可以用servletcontext域实现共享

2件事情:1 是一个容器

2 servletcontext 域说明这个容器的作用范围

请求转发:1次

重定向: 2次请求

每个web应用都会有个servletcontext 停掉的时候会销毁

配置文件: xml(数据有关系) property(数据没关系)

通过ServletContext读取资源文件

过servletContext的getRealPath得到资源的绝对路径后,再通过传统流读取资源文件

可以得到资源的名称 比较好

              String path = this.getServletContext().getRealPath("/WEB-INF/classes/db.properties");

              String filename = path.substring(path.lastIndexOf("\\")+1);//得到资源名称的方法

             

              System.out.println("当前读取到的资源名称是:" + filename); 

             

              FileInputStream in = new FileInputStream(path);

              Properties props = new Properties();  //map

              props.load(in);

             

              String url = props.getProperty("url");

              String username = props.getProperty("username");

              String password = props.getProperty("password");

             

              System.out.println("当前读取到的资源数据是:");

              System.out.println(url);

              System.out.println(username);

              System.out.println(password);

如果读取资源文件的程序不是servlet的话,就只能通过类装载器去读了  类装载的方式读不到更新的数据

/*

              //以下代码虽然可以读取资源文件的数据,但是无法获取更新后的数据 类装载的方式只装载一次

              Properties dbconfig = new Properties();

              InputStream in = UserDao.class.getClassLoader().getResourceAsStream("db.properties");

              dbconfig.load(in);

              System.out.println(dbconfig.getProperty("url"));*/

//通过类装载的方式得到资源文件的位置,再通过传统方式读取资源文件的数据,这样可以读取到更新后的数据  需要更新 就用这个

              String path = UserDao.class.getClassLoader().getResource("db.properties").getPath();

              FileInputStream in = new FileInputStream(path);

              Properties dbconfig = new Properties();

              dbconfig.load(in);

              System.out.println(dbconfig.getProperty("url"));

在客户端缓存Servlet的输出

l  对于不经常变化的数据,在servlet中可以为其设置合理的缓存时间值,以避免浏览器频繁向服务器发送请求,提升服务器的性能

range头实现断点下载

阅读更多
想对作者说点什么? 我来说一句

没有更多推荐了,返回首页

加入CSDN,享受更精准的内容推荐,与500万程序员共同成长!
关闭
关闭