Servlet的细节问题

1.一个已经注册的Servlet可以被多次映射即:

  1. <servlet>
  2.   <!-- servlet的注册名 -->
  3.   <servlet-name>MyServlet1</servlet-name>
  4.   <!-- servlet类的全路径(包名+类名) -->
  5.   <servlet-class>com.hsp.servlet.MyServlet1</servlet-class>
  6. </servlet>
  7. <!-- 对一个已经注册的servlet的映射 -->
  8. <servlet-mapping>
  9.   <!-- servelt的注册名 -->
  10.   <servlet-name>MyServlet1</servlet-name>
  11.   <!-- servlet的访问路径 -->
  12.   <url-pattern>/MyServlet1</url-pattern>
  13. </servlet-mapping>
  14. <servlet-mapping>
  15.   <servlet-name>MyServlet1</servlet-name>
  16.   <url-pattern>/winner</url-pattern>
  17. </servlet-mapping>

2.当映射一个servlet时候,可以多层 比如

<url-pattern>/servlet/index.html</url-pattern> ok

从这里还可以看出,后缀名是html不一定就是html,可能是假象.

3.把servlet映射到URL中时可以使用*通配符,但是只能有两种方式。

有两种格式:

第一种格式  *.扩展名  比如 *.do,*.ss

第二种格式  以/开头,同时以 /* 结尾.比如/*(所有的请求都跳转到某个地方,如要把整个网站关闭)

/news/*(我们要把门户网站的新闻频道关闭,,所有对新闻频道的访问都跳转到某个页面)

通配符练习题:

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

在匹配的时候,要参考的标准:

(1)看谁的匹配度高,谁就被选择

(2)*.do 的优先级最低

实际应用:现在我们要把整个网站关闭,我们自己写一个servlet,如CloseServlet,向页面输出out.println(‘对不起,该网站暂时关闭’)。然后配置web.xml:

  1. <servlet>
  2.   <servlet-name>CloseServlet</servlet-name>
  3.   <servlet-class>com.....CloseServlet</servlet-class>
  4. </servlet>
  5. <servlet-mapping>
  6.   <servlet-name>CloseServlet</servlet-name>
  7.   <url-pattern>/*</url-pattern>
  8. </servlet-mapping>

对这个网站的所有请求都由CloseServlet处理。

将原来web.xml拷贝一份,重命名为web.xml.bak,网站恢复之后再改回来!

4.Servlet单例问题

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

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

当Servlet被第一次访问后,就被加载到内存,以后该实例对各个请求服务.即在使用中是单例.

因为Servlet是单例,因此会出现线程安全问题: 比如:售票系统. 如果不加同步机制,则会出现问题:

怎么证明是单例呢?

  1. public class MyServlet1 extends HttpServlet {
  2. int i = 0;
  3. protected void doGet(HttpServletRequest request, HttpServletResponse response)
  4. throws ServletException, IOException {
  5. this.doPost(request, response);
  6. }
  7. protected void doPost(HttpServletRequest request, HttpServletResponse response)
  8. throws ServletException, IOException {
  9. i++;
  10. response.getWriter().println("hello, i="+i);
  11. }
  12. }

i的值会一直增加!多个请求增加的是同一个i。

售票可能出现的问题:

  1. public class MyServlet1 extends HttpServlet {
  2. int ticket = 2;
  3. protected void doGet(HttpServletRequest request, HttpServletResponse response)
  4. throws ServletException, IOException {
  5. this.doPost(request, response);
  6. }
  7. protected void doPost(HttpServletRequest request, HttpServletResponse response)
  8. throws ServletException, IOException {
  9. if(ticket > 0){
  10. System.out.println("你买到票了");
  11. try {
  12. //线程休眠用来模拟成千上万个人同时并发,因为一个售票系统会有很多人同时访问!
  13. Thread.sleep(10 * 1000);
  14. } catch (InterruptedException e) {
  15. // TODO Auto-generated catch block
  16. e.printStackTrace();
  17. }
  18. //先休眠再减!!买完票还没来得及减另一个人就来买了!
  19. ticket--;
  20. }else{
  21. System.out.println("你买不到票了");
  22. }
  23. }
  24. }

解决:

(1)如果一个变量需要多个用户共享,则应当在访问该变量的时候,加同步机制

  1. synchronized (对象){
  2. //同步代码
  3. }
  1. public class MyServlet1 extends HttpServlet {
  2. int ticket = 2;
  3. protected void doGet(HttpServletRequest request, HttpServletResponse response)
  4. throws ServletException, IOException {
  5. this.doPost(request, response);
  6. }
  7. protected void doPost(HttpServletRequest request, HttpServletResponse response)
  8. throws ServletException, IOException {
  9. synchronized (this) {
  10. if(ticket > 0){
  11. System.out.println("你买到票了");
  12. try {
  13. //线程休眠用来模拟成千上万个人同时并发,因为一个售票系统会有很多人同时访问!
  14. Thread.sleep(10 * 1000);
  15. } catch (InterruptedException e) {
  16. // TODO Auto-generated catch block
  17. e.printStackTrace();
  18. }
  19. //先休眠再减!!买完票还没来得及减另一个人就来买了!
  20. ticket--;
  21. }else{
  22. System.out.println("你买不到票了");
  23. }
  24. }
  25. }
  26. }

(2)如果一个变量不需要共享,则直接在doGet()或者doPost()定义,这样不会存在线程安全问题.

  1. public class MyServlet1 extends HttpServlet {
  2. protected void doGet(HttpServletRequest request, HttpServletResponse response)
  3. throws ServletException, IOException {
  4. this.doPost(request, response);
  5. }
  6.  
  7. protected void doPost(HttpServletRequest request, HttpServletResponse response)
  8. throws ServletException, IOException {
  9. int j = 0;
  10. j ++;
  11. response.getWriter().println("j=" + j);
  12. }
  13. }

无论谁请求j的值都是1.

5.servlet 中的 <load-on-startup> 配置

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

需求: 当我们的网站启动的时候,可能会要求初始化一些数据,(比如创建临时表), 在比如:

我们的网站有一些要求定时完成的任务[ 定时写日志,定时备份数据.. 定时发送邮件..]

解决方法: 可以通过 <load-on-startup> 配合线程知识搞定.

先说明<load-on-startup>: 通过配置<load-on-startup> 我们可以指定某个Servlet 自动创建.

<load-on-startup>1</load-on-startup>

这个1是指servlet被初始化的顺序。

对于一个大型的网站来说,初始化的servlet可能不止一个,它们的相对执行顺序用数字表示

任务:web应用启动时创建一个后台线程,定时完成某些任务(每隔十秒发一封邮件)。

我们来模拟一个定时发送电子邮件的功能:这个功能应该在网站启动的时候就开始

实现思路:

sendEmailTable

id                  content                 sendtime

1                  "hello"                   2011-11-11 20:11

2                   "hello2"                2012-11-11 10:00

  1. public class SendEmailThread extends Thread{
  2. @Override
  3. public void run() {
  4. int i=0;
  5. try {
  6. //这个事应该在一直执行。
  7. while(true){
  8. //每休眠一分钟,就去扫表sendmail, 看看那份信件应当被发出
  9. Thread.sleep(10*1000);
  10. System.out.println("发出 第"+(++i)+"邮件");//javamail
  11. }
  12. } catch (Exception e) {
  13. e.printStackTrace();
  14. // TODO: handle exception
  15. }
  16. }
  17. }
  1. public class MyInitServlet extends HttpServlet {
  2. private static final long serialVersionUID = 1L;
  3.  
  4. public void init() throws ServletException {
  5. // Put your code here
  6. System.out.println("MyInitServlet1 的init被调用..");
  7. //完成一些初始化任务
  8. System.out.println("创建数据库,表,读取参数");
  9. //创建一个线程
  10. SendEmailThread sendEmailThread=new SendEmailThread();
  11. sendEmailThread.start();
  12. }
  13. }

ServletConfig对象

该对象主要用于 读取 servlet的配置信息.

  1. <servlet>
  2. <servlet-name>ServletConfigTest</servlet-name>
  3. <servlet-class>com.hsp.servlet.ServletConfigTest</servlet-class>
  4. <!-- 这里可以给servlet配置信息,这里配置的信息,只能被该servlet 读取 -->
  5. <init-param>
  6.   <param-name>encoding</param-name>
  7.   <param-value>utf-8</param-value>
  8. </init-param>
  9. </servlet>

如何使用

  1. String encoding = this.getServletConfig().getInitParameter("encoding");
  2. response.setCharacterEncoding(encoding);

servlet可以指定编码方式

补充说明:这种配置参数的方式,只能被某个Servlet独立使用.如希望让所有的Servlet都去读取某个参数,这样配置:

  1. <!-- 如果这里配置参数,可被所有servlet读取 -->
  2. <!--
  3. <context-param>
  4. <param-name></param-name>
  5. <param-value></param-value>
  6. </context-param>
  7. -->
  1. 如果要把所有的参数都读取,则使用 如下方法
  2. Enumeration<String> names=this.getServletConfig().getInitParameterNames();
  3.  
  4. while(names.hasMoreElements()){
  5.   String name=names.nextElement();
  6.   System.out.println(name);
  7.   System.out.println(this.getServletConfig().getInitParameter(name));
  8. }

Servlet的一些细节问题的更多相关文章

  1. servlet中的细节

    Get方法有大小限制:1024个字符.这些信息使用 Query_String头传递,并通过Query_String环境变量访问.Post方法:请求体信息使用FromData头传递.读取所有表单参数:g ...

  2. Servlet的一些细节(2)

    1. Servlet的创建时间 Servlet是不能单独运行,调用它的叫做Servlet引擎,或者叫做web服务器 针对客户端的多长Servlet请求,通常情况下,服务器只会创建一个Servlet实例 ...

  3. Servlet的一些细节(1)

    1.  Servlet程序必须映射到一个URL地址 由于客户端是通过URL访问web服务器资源,所以Servlet程序必须映射到一个URL地址.这个工作在web.xml文件中使用<servlet ...

  4. [Java.Web] Servlet 的一些细节

    本文来自 传智播客视频PPT 1. 由于客户端是通过 URL 地址访问 web 服务器中的资源,所以 Servlet 程序若想被外界访问,必须把 servlet 程序映射到一个 URL 地址上,这个工 ...

  5. Servlet的一些细节

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

  6. 2016-2-1 Servlet细节

    Servlet的一些细节(韩顺平老师视频讲解)(1)由于客户端是通过URL地址访问web服务器中的资源,所以Servlet程序想要被外界访问,必须把servlet程序映射到一个URL地址上.这个工作在 ...

  7. JavaWeb -- Servlet运行过程 和 细节

    Servlet的运行过程 lServlet程序是由WEB服务器调用,web服务器收到客户端的Servlet访问请求后: ①Web服务器首先检查是否已经装载并创建了该Servlet的实例对象.如果是,则 ...

  8. Servlet映射细节

    Servlet的映射细节: 1):一个Servlet程序(Web组件),可以配置多个<url-pattern>,表示一个Servlet有多个资源名称. <servlet-mappin ...

  9. java web学习总结(五) -------------------servlet开发(一)

    一.Servlet简介 Servlet是sun公司提供的一门用于开发动态web资源的技术. Sun公司在其API中提供了一个servlet接口,用户若想用发一个动态web资源(即开发一个Java程序向 ...

随机推荐

  1. java实现的一个maven多模块项目自动生成工具

    平时在做spring mvc web新项目时,都需要自己去搭建spring mvc的项目框架,包括基本pom 依赖引入,基本配置文件(web.xml,spring-mvc.xml,数据库配置文件等等) ...

  2. eclispe 出现超内纯错误

    刚开始以为只要修改tomcat的最大最小内存就可以,结果还是报错,后来才懂需要在eclipse.ini文件中修改 -Xms256m-Xmx512m的值改大些,增加虚拟机运行的内存空间 刚开始最小值只有 ...

  3. 浅谈css中的position属性

    我觉得吧,css如果不考虑浏览器的兼容问题的话,最让人头疼的应该就是position了,反正我是这么觉得的,为了能基本上搞清楚position的几种情况,我找了一些资料,做了一个小实验,下面是实验的过 ...

  4. Linux C SMTP POP3 极简陋邮件客户端

    以前以为协议非常高神,但做了这个之后发现还好,没想象的那么艰难. 先要了解邮件的原理 再者就是上面2协议 可以先用telnet测试一下,以初步了解那2协议:http://hi.baidu.com/34 ...

  5. 如何在IOS开发中在自己的framework中添加.bunble文件

    今天就跟大家介绍一下有关,如何在IOS开发中在自己的framework中添加.bunble文件,该文章我已经在IOS教程网(http://ios.662p.com)发布过来,个人觉得还是对大家有帮助的 ...

  6. 服务器 tfs不提供 TeamFoundation服务。基础连接已经关闭

    服务器 tfs(服务器名或url)不提供 TeamFoundation服务.基础连接已经关闭,发送时发生错误.TFS突然间连接不上到,到服务器上配置团队项目的组成员资格提示这样的错误,客户端连接的时候 ...

  7. 在.NET连接MySQL以及封装好的MySQLHelper.cs

    1.首先上MySQL网站下驱动:http://www.mysql.com/products/connector/ 2.安装下载的安装包 3.我们在Visual Studio里创建一个Web Appli ...

  8. [转]SET NOCOUNT ON

    ref: http://www.cnblogs.com/jayleke/archive/2010/07/10/1774758.html 在存储过程,触发器中,经常用到SET NOCOUNT ON: 作 ...

  9. mcrypt加密与解密

    $key = 'test'; $result_array = array('name' => 'ta', 'age' => 28); $str = encode($result_array ...

  10. 【Delphi】无标题移动窗体

    procedure TForm1.FormMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Inte ...