Servlet一些细节:
1. 由于客户端是通过URL地址访问web服务器中的资源,所以Servlet程序若想被外界访问的话,必须把servlet程序映射到一个URL地址上,这个工作在web.xml文件中使用<servlet>元素和<servlet-mapping>元素完成。
2. <servlet>元素用于注册Servlet,它包含有两个主要的子元素:<servlet-name>和<servlet-class>,分别用于设置Servlet的注册名称和Servlet的完整类名。
3. 一个<servlet-mapping>元素用于映射一个已经注册的Servlet的一个对外的访问路径,它包含两个子元素:<servlet-name>和<url-pattern>,分别用于指定Servlet的注册名称和Servlet的对外访问路径。
注意:对于一个已经注册的Servlet可以被多次映射。
4. 在对servlet进行映射的时候,可以使用通配符
有两种格式:
第一种格式:*.扩展名 比如说*.do *.ss
第二种格式:以/开头并且以/*结尾 比如/* /news/*
在通配符进行匹配的时候,要参考的标准
(1) 看谁的匹配度高,谁就被选中
(2) *.do的优先级最低
5. Servlet单例问题
当Servlet被第一次访问后,就被加载到内存,以后该实例对各个请求服务即在使用中是单例的。服务器只会创建一个Servlet实例对象,也就是说该Servlet实例对象一旦创建就会驻留在内存中,为后续的其他请求服务,直到web容器退出或者reload该web应用程序,servlet对象才会被销毁。
因为Servlet是单例,因此会出现线程安全问题,比如:
售票系统,如果不加同步机制,则会出现问题。
给大家的一个原则:
(1) 如果一个变量需要多个用户共享,则应当在访问该变量的时候,加同步机制synchronize(对象){
//同步代码
}
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html");
PrintWriter out = response.getWriter();
int j = 0;
j++;
out.println("hello "+new java.util.Date().toString()+" j="+j);
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()中定义即可。这样不会存在线程安全问题。
6. servlet中的<load-on-startup>配置
需求:当我们的网站启动的时候,可能会要求初始化一些数据,(比如创建临时表),再比如:
我们的网站要有一些定时完成的任务【定时写日志,定时备份数据库…定时发送邮件】
解决方法:可以通过<load-on-startup>配合线程相关知识搞定。
先说明<load-on-startup>:通过配置它,我么可以指定某个Servlet自动创建。
<servlet>
<description>This is the description of my J2EE component</description>
<display-name>This is the display name of my J2EE component</display-name>
<servlet-name>MyInitServlet1</servlet-name>
<servlet-class>com.sgr.servlet.MyInitServlet1</servlet-class>
<!--1表示该Servlet被init的顺序-->
<load-on-startup>2</load-on-startup>
</servlet>
<servlet>
<description>This is the description of my J2EE component</description>
<display-name>This is the display name of my J2EE component</display-name>
<servlet-name>MyInitServlet2</servlet-name>
<servlet-class>com.sgr.servlet.MyInitServlet2</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
配置好<load-on-startup>之后就可以在init函数当中进行相关代码的编写了,例如我在另外一个包当中写了一个简单的模拟发邮件java程序,代码如下:
package com.sgr.model;
public class SendEmailThread extends Thread{
public void run() {
// TODO Auto-generated method stub
int i = 0;
try {
//每休眠一分钟,就去扫描sendmail,看看哪封信应当被发出
while(true) {
Thread.sleep(10*1000);
System.out.println("发出第"+(++i)+"邮件");//javamail技术
}
} catch (Exception e) {
// TODO: handle exception
e.printStackTrace();
}
}
}
之后我在我之前写好的Servlet的init函数当中写一些需要初始化而做的工作。代码如下:
public void init() throws ServletException {
// Put your code here
System.out.println("MyInitServlet1的Init被调用");
//完成初始化任务
System.out.println("创建数据库,表,读取");
//创建一个线程
SendEmailThread sendEmailThread = new SendEmailThread();
sendEmailThread.start();
}