生命周期
当servlet被部署在应用服务器中(应用服务器中用于管理Java组件的部分被抽象成为)以后,由容器控制servlet的生命周期。除非特殊指定,否则在容器启动的时候,servlet是不会被加载的,servlet只会在第一次请求的时候被加载和实例化。servlet一旦被加载,一般不会从容器中删除,直至应用服务器关闭或重新启动。但当容器做内存回收动作时,servlet有可能被删除。也正是因为这个原因,第一次访问servlet所用的时间要大大多于以后访问所用的时间。
servlet在服务器的运行生命周期为,在第一次请求(或其实体被内存垃圾回收后再被访问)时被加载并执行一次初始化方法,跟着执行正式运行方法,之后会被常驻并每次被请求时直接执行正式运行方法,直到服务器关闭或被清理时执行一次销毁方法后实体销毁。
Servlet自动加载
init做初始化的操作,非常消耗时间的。在<servlet>标签下 <load-on-startup>3</load-on-startup> * 值是正整数 * 如果值越小,优先级越高。Servlet访问URL映射配置
ServletDemo1 gacl.servlet.study.ServletDemo1 ServletDemo1 /servlet/ServletDemo1
同一个Servlet可以被映射到多个URL上
如果某个Servlet的映射路径仅仅为一个正斜杠(/),那么这个Servlet就成为当前Web应用程序的缺省Servlet。
凡是在web.xml文件中找不到匹配的<servlet-mapping>元素的URL,它们的访问请求都将交给缺省Servlet处理,也就是说,缺省Servlet用于处理所有其他Servlet都不处理的访问请求。
Servlet映射匹配问题
配置虚拟路径(访问路径)
完全路径匹配 以/开头的 目录匹配 以/开头的 /* 扩展名匹配 不能以/开头的 *.do *.action 优先级:完全路径匹配 > 目录匹配 > 扩展名匹配(*****)WEB开发中路径的问题
相对路径 * 一个文件相对于另一个文件的位置的关系。 * 不能以/开头 写法: ./demo demo ../demo * 访问1.html: http://localhost/day09/1.html * 访问demo5: http://localhost/day09/demo5 * 从1.html中去访问demo5:./demo5 demo5 * 访问2.html: http://localhost/day09/html/2.html * 访问demo5: http://localhost/day09/demo5 * 从2.html访问demo5:../demo5绝对路径(推荐使用) * 以/开头的 * 访问demo5: http://localhost/day09/demo5 * 从1.html使用绝对路径访问demo5:http://localhost/day09/demo5 * 简写方式:/day09/demo5 * 客户端绝对路径 * /day09/demo5 需要写项目名 * 服务器绝对路径 * /demo5 不能写项目名ServletContext对象(域对象)
* 定义:WEB容器在启动时,它会为每个WEB应用程序都创建一个对应的ServletContext对象,它代表当前web应用。 一个WEB应用对应一个ServletContext对象 一个WEB应用下有多个Servlet程序 所有的servlet程序都共享同一个ServletContext对象 * 作用: * 获取WEB应用全局初始化参数 * 在web.xml中配置String getInitParameter(String name) getInitParameterNames() encoding GBK
实现数据的共享(*****)
void setAttribute(String name, Object object) 存入数据void removeAttribute(String name) 删除数据Object getAttribute(String name) 获取数据
读取资源文件(*****)
InputStream getResourceAsStream(String path) 通过文件的地址获取输入流String getRealPath(String path) 通过文件的地址获取文件的绝对磁盘路径
获取Web应用的初始化参数
Servlet的线程安全问题
当多个客户端并发访问同一个Servlet时,web服务器会为每一个客户端的访问请求创建一个线程,并在这个线程上调用Servlet的service方法,因此service方法内如果访问了同一个资源的话,就有可能引发线程安全问题。
针对Servlet的线程安全问题,Sun公司是提供有解决方案的:让Servlet去实现一个SingleThreadModel接口,如果某个Servlet实现了SingleThreadModel接口,那么Servlet引擎将以单线程模式来调用其service方法。
查看Sevlet的API可以看到,SingleThreadModel接口中没有定义任何方法和常量,在Java中,把没有定义任何方法和常量的接口称之为标记接口,经常看到的一个最典型的标记接口就是"Serializable",这个接口也是没有定义任何方法和常量的,标记接口在Java中有什么用呢?主要作用就是给某个对象打上一个标志,告诉JVM,这个对象可以做什么,比如实现了"Serializable"接口的类的对象就可以被序列化,还有一个"Cloneable"接口,这个也是一个标记接口,在默认情况下,Java中的对象是不允许被克隆的,就像现实生活中的人一样,不允许克隆,但是只要实现了"Cloneable"接口,那么对象就可以被克隆了。让Servlet实现了SingleThreadModel接口,只要在Servlet类的定义中增加实现SingleThreadModel接口的声明即可。
对于实现了SingleThreadModel接口的Servlet,Servlet引擎仍然支持对该Servlet的多线程并发访问,其采用的方式是产生多个Servlet实例对象,并发的每个线程分别调用一个独立的Servlet实例对象。 实现SingleThreadModel接口并不能真正解决Servlet的线程安全问题,因为Servlet引擎会创建多个Servlet实例对象,而真正意义上解决多线程安全问题是指一个Servlet实例对象被多个线程同时调用的问题。事实上,在Servlet API 2.4中,已经将SingleThreadModel标记为Deprecated(过时的)。