共计 3937 个字符,预计需要花费 10 分钟才能阅读完成。
一、Servlet 与线程安全
因为一个类型的 Servlet 只有一个实例对象,那么就有可能会现在某一时刻一个 Servlet 同时处理多个请求,那么Servlet 是否为线程安全的呢?
答案是:“不是线程安全的”。这说明 Servlet 的工作效率很高,但也存在线程安全问题!
所以我们不应该在 Servlet 中便宜创建成员变量,因为可能会存在一个线程对这个成员变量进行写操作,另一个线程对这个成员变量进行读操作。
注意:
- 不要在 Servlet 中创建成员!创建局部变量即可!
- 可以创建无状态成员!
- 可以创建有状态的成员,但状态必须为只读的!
创建局部变量
public class DServlet extends GenericServlet {@Override
public void init() throws ServletException {System.out.println("哈哈哈~我来也!");
}
@Override
public void service(ServletRequest arg0, ServletResponse arg1)
throws ServletException, IOException {// 创建局部变量
Object obj = new Object();
System.out.println("hello world");
}
@Override
public void destroy() {System.out.println("destory()");
}
}
创建无状态成员
public class User {public String getName() {return "zhangSan";
}
}
public class DServlet extends GenericServlet {// 创建无状态成员
User u = new User();
@Override
public void init() throws ServletException {System.out.println("哈哈哈~我来也!");
}
@Override
public void service(ServletRequest arg0, ServletResponse arg1)
throws ServletException, IOException {System.out.println("hello world");
System.out.println(user.getName());
}
@Override
public void destroy() {System.out.println("destory()");
}
}
创建有状态的成员,状态为只读
public class User {private String name = "zhangSan";// 属性就是状态
public String getName() {return name;
}
public void hello() {System.out.println("hello");
}
}
public class DServlet extends GenericServlet {// 创建有状态成员, 但状态为只读
User u = new User();
@Override
public void init() throws ServletException {System.out.println("哈哈哈~我来也!");
}
@Override
public void service(ServletRequest arg0, ServletResponse arg1)
throws ServletException, IOException {System.out.println("hello world");
System.out.println(u.getName());
}
@Override
public void destroy() {System.out.println("destory()");
}
}
二、让服务器在启动时就创建 Servlet
默认情况下,服务器会在某个 Servlet 第一次收到请求时创建它。也可以在 web.xml 中对 Servlet 进行配置,使服务器启动时就创建 Servlet。
<servlet>
<servlet-name>hello1</servlet-name>
<servlet-class>
com.tyschool.servlet.Hello1Servlet
</servlet-class>
<load-on-startup>0</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>hello1</servlet-name>
<url-pattern>/hello1</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>hello2</servlet-name>
<servlet-class>
com.tyschool.servlet.Hello2Servlet
</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>hello2</servlet-name>
<url-pattern>/hello2</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>hello3</servlet-name>
<servlet-class>
com.tyschool.servlet.Hello3Servlet
</servlet-class>
<load-on-startup>2</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>hello3</servlet-name>
<url-pattern>/hello3</url-pattern>
</servlet-mapping>
注意:在
在
三、访问路径
- 可以在
中给出多个 ,例如:
<servlet-mapping>
<servlet-name>AServlet</servlet-name>
<url-pattern>/AServlet</url-pattern>
<url-pattern>/BServlet</url-pattern>
</servlet-mapping>
那么这说明一个 Servlet 绑定了两个 URL,无论访问 /AServlet 还是 /BServlet,访问的都是 AServlet
- 还可以在
中使用通配符,所谓通配符就是星号 *,星号可以匹配任何 URL 前缀或后缀,使用通配符可以命名一个 Servlet 绑定一组 URL,例如:
<url-pattern>/servlet/*<url-patter>
路径匹配
该路径可以匹配:/servlet/a、/servlet/b,都匹配 /servlet/;<url-pattern>*.do </url-pattern>
扩展名匹配
该路径可以匹配:/abc/def/ghi.do、/a.do,都匹配.do;<url-pattern>/*<url-pattern>
该路径可以匹配:匹配所有 URL;
注意:
通配符要么为前缀,要么为后缀,不能出现在 URL 中间位置,也不能只有通配符。
例如:
<url-pattern>/*.do<url-pattern>
就是错误的,因为星号出现在 URL 的中间位置上了。
例如:
<url-pattern>*.*<url-pattern>
也是不对的,因为 一个 URL 中最多只能出现一个通配符。
注意
通配符是一种模糊匹配 URL 的方式,如果存在更具体的
例如:
<servlet>
<servlet-name>hello1</servlet-name>
<servlet-class>
com.tyschool.servlet.Hello1Servlet
</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>hello1</servlet-name>
<url-pattern>/servlet/hello1</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>hello2</servlet-name>
<servlet-class>
com.tyschool.servlet.Hello2Servlet
</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>hello2</servlet-name>
<url-pattern>/servlet/*</url-pattern>
</servlet-mapping>
当访问路径为 http://localhost:8080/hello/servlet/hello1 时,因为访问路径即匹配 hello1 的