Servlet的一些细节问题
Servlet的细节问题
1.一个已经注册的Servlet可以被多次映射即:
<servlet> <!-- servlet的注册名 --> <servlet-name>MyServlet1</servlet-name> <!-- servlet类的全路径(包名+类名) --> <servlet-class>com.hsp.servlet.MyServlet1</servlet-class> </servlet> <!-- 对一个已经注册的servlet的映射 --> <servlet-mapping> <!-- servelt的注册名 --> <servlet-name>MyServlet1</servlet-name> <!-- servlet的访问路径 --> <url-pattern>/MyServlet1</url-pattern> </servlet-mapping> <servlet-mapping> <servlet-name>MyServlet1</servlet-name> <url-pattern>/winner</url-pattern> </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:
<servlet> <servlet-name>CloseServlet</servlet-name> <servlet-class>com.....CloseServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>CloseServlet</servlet-name> <url-pattern>/*</url-pattern> </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是单例,因此会出现线程安全问题: 比如:售票系统. 如果不加同步机制,则会出现问题:
怎么证明是单例呢?
public class MyServlet1 extends HttpServlet { int i = 0; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request, response); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { i++; response.getWriter().println("hello, i="+i); } }
i的值会一直增加!多个请求增加的是同一个i。
售票可能出现的问题:
public class MyServlet1 extends HttpServlet { int ticket = 2; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request, response); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { if(ticket > 0){ System.out.println("你买到票了"); try { //线程休眠用来模拟成千上万个人同时并发,因为一个售票系统会有很多人同时访问! Thread.sleep(10 * 1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } //先休眠再减!!买完票还没来得及减另一个人就来买了! ticket--; }else{ System.out.println("你买不到票了"); } } }
解决:
(1)如果一个变量需要多个用户共享,则应当在访问该变量的时候,加同步机制
synchronized (对象){ //同步代码 }
public class MyServlet1 extends HttpServlet { int ticket = 2; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request, response); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { synchronized (this) { if(ticket > 0){ System.out.println("你买到票了"); try { //线程休眠用来模拟成千上万个人同时并发,因为一个售票系统会有很多人同时访问! Thread.sleep(10 * 1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } //先休眠再减!!买完票还没来得及减另一个人就来买了! ticket--; }else{ System.out.println("你买不到票了"); } } } }
(2)如果一个变量不需要共享,则直接在doGet()或者doPost()定义,这样不会存在线程安全问题.
public class MyServlet1 extends HttpServlet { protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request, response); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { int j = 0; j ++; response.getWriter().println("j=" + j); } }
无论谁请求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
public class SendEmailThread extends Thread{ @Override public void run() { int i=0; try { //这个事应该在一直执行。 while(true){ //每休眠一分钟,就去扫表sendmail, 看看那份信件应当被发出 Thread.sleep(10*1000); System.out.println("发出 第"+(++i)+"邮件");//javamail } } catch (Exception e) { e.printStackTrace(); // TODO: handle exception } } }
public class MyInitServlet extends HttpServlet { private static final long serialVersionUID = 1L; public void init() throws ServletException { // Put your code here System.out.println("MyInitServlet1 的init被调用.."); //完成一些初始化任务 System.out.println("创建数据库,表,读取参数"); //创建一个线程 SendEmailThread sendEmailThread=new SendEmailThread(); sendEmailThread.start(); } }
ServletConfig对象
该对象主要用于 读取 servlet的配置信息.
<servlet> <servlet-name>ServletConfigTest</servlet-name> <servlet-class>com.hsp.servlet.ServletConfigTest</servlet-class> <!-- 这里可以给servlet配置信息,这里配置的信息,只能被该servlet 读取 --> <init-param> <param-name>encoding</param-name> <param-value>utf-8</param-value> </init-param> </servlet>
如何使用
String encoding = this.getServletConfig().getInitParameter("encoding"); response.setCharacterEncoding(encoding);
servlet可以指定编码方式
补充说明:这种配置参数的方式,只能被某个Servlet独立使用.如希望让所有的Servlet都去读取某个参数,这样配置:
<!-- 如果这里配置参数,可被所有servlet读取 --> <!-- <context-param> <param-name></param-name> <param-value></param-value> </context-param> -->
如果要把所有的参数都读取,则使用 如下方法 : Enumeration<String> names=this.getServletConfig().getInitParameterNames(); while(names.hasMoreElements()){ String name=names.nextElement(); System.out.println(name); System.out.println(this.getServletConfig().getInitParameter(name)); }
Servlet的一些细节问题的更多相关文章
- servlet中的细节
Get方法有大小限制:1024个字符.这些信息使用 Query_String头传递,并通过Query_String环境变量访问.Post方法:请求体信息使用FromData头传递.读取所有表单参数:g ...
- Servlet的一些细节(2)
1. Servlet的创建时间 Servlet是不能单独运行,调用它的叫做Servlet引擎,或者叫做web服务器 针对客户端的多长Servlet请求,通常情况下,服务器只会创建一个Servlet实例 ...
- Servlet的一些细节(1)
1. Servlet程序必须映射到一个URL地址 由于客户端是通过URL访问web服务器资源,所以Servlet程序必须映射到一个URL地址.这个工作在web.xml文件中使用<servlet ...
- [Java.Web] Servlet 的一些细节
本文来自 传智播客视频PPT 1. 由于客户端是通过 URL 地址访问 web 服务器中的资源,所以 Servlet 程序若想被外界访问,必须把 servlet 程序映射到一个 URL 地址上,这个工 ...
- Servlet的一些细节
由于客户端是通过URL地址访问web服务器的中的资源的,所以Servlet程序若想被外界访问,必须把servlet程序映射到一个URL地址上,这个工作在web.xml文件中使用<servlet& ...
- 2016-2-1 Servlet细节
Servlet的一些细节(韩顺平老师视频讲解)(1)由于客户端是通过URL地址访问web服务器中的资源,所以Servlet程序想要被外界访问,必须把servlet程序映射到一个URL地址上.这个工作在 ...
- JavaWeb -- Servlet运行过程 和 细节
Servlet的运行过程 lServlet程序是由WEB服务器调用,web服务器收到客户端的Servlet访问请求后: ①Web服务器首先检查是否已经装载并创建了该Servlet的实例对象.如果是,则 ...
- Servlet映射细节
Servlet的映射细节: 1):一个Servlet程序(Web组件),可以配置多个<url-pattern>,表示一个Servlet有多个资源名称. <servlet-mappin ...
- java web学习总结(五) -------------------servlet开发(一)
一.Servlet简介 Servlet是sun公司提供的一门用于开发动态web资源的技术. Sun公司在其API中提供了一个servlet接口,用户若想用发一个动态web资源(即开发一个Java程序向 ...
随机推荐
- TCL_事务控制语言
TCL transaction 事务 -- DML 定义为把一连串的操作作为单个逻辑工作单元处理 ----- 例如:银行转账 ...
- nodejs remote链接mysql数据库总结
nodejs链接远端mysql,这个折腾了一个上午才搞定.本以为,直接使用就OK了,但是发现不行,后来查阅各种资料后,终于找到了方法. nodejs链接远端数据库主要分为几个步骤: 1)安装node- ...
- C++ 读取 pcap文件.
http://blog.csdn.net/haolipengzhanshen/article/details/51854853 1.了解下pcap文件的结构 2.定义pcap文件头部结构体pcapFi ...
- automake使用
antuomake 流程图: http://blog.csdn.net/houwei544/article/details/8185916 这个教程不错 https://www.ibm.com/dev ...
- Google的小秘密
google有计算器的功能,例如在google中搜索25*25.lg(13)等,看会出现什么样的结果. http://www.google.com/microsoft 微软风格的入口 http: ...
- System.UnauthorizedAccessException: 拒绝访问 temp 目录。用来运行 XmlSerializer 的标识“NT AUTHORITY\NETWORK SERVICE”没有访问 temp 目录的足够权限。CodeDom 将使用进程正在使用的用户帐户进行编译,这样,如
解决方案:IIS的应用程序池权限不够,应用程序给localsystem账号权限即可. 以客户的服务器系统2003sp2为例,修改步骤如下: 控制面板---管理工具--Internet 信息服务(IIS ...
- 【原】Oracle查询指定表里的触发器
select * from all_triggers WHERE table_name='表名'
- AngularJS(8)-指令directive
AngularJS 提供了很多内置的指令,你可以使用它们来为你的应用添加功能. 诸如这些: 此外,你可以使用模块来为你应用添加自己的指令: 运行结果:
- ES6学习笔记(三)
ES6加强了对Unicode的支持,并且扩展了字符串对象. 1.字符的Unicode表示法 JavaScript允许采用\uxxxx形式表示一个字符,其中"xxxx"表示字符的码点 ...
- MySQL Connector Net连接vs2012问题
最近做一.NET项目,数据库用到MySQL,可是在VS2012连接数据库是遇到问题,提示:Authentication with old password no longer supported, u ...