JavaEE5 Tutorial_Servlet

Web资源:web组件,静态web文件如图片

Web程序:可发布的Web资源集合

 

Web程序根目录下有个web-inf文件夹,如果只有jsp和静态资源,里面可以没有web.xml

根目录下可以直接放东西,也可以新建自定义目录

 

自动部署功能在开发完上线后要记得关闭,负责一自动重新部署,所有会话将失效

 

包javax.servlet和javax.servlet.http提供了servlet的接口和类
所有的servlet类都必须实现servlet接口,它定义了servlet的生命周期

父类可以是GenericServlet类或HttpServlet类

Web容器控制着Servlet的生命周期,当请求调用某个Servlet时
找到这个类,实例化,init()初始化,都只有一次
调用service(),传递request和response,可以反复
最后调用destroy(),一次

在生命周期中某些事件发生时,可以绑定一些监听器做一些处理
可以监听ServletContext,Session,Request

可以指定某个具体异常到某个页面处理

Web组件之间共享信息可以通过私有的javaBean属性,公开的域对象的属性,干脆数据库,或者其他Web资源

域对象有四种:
Javax.servlet.ServletContext
Javax.servlet.http.HttpSession
javax.servlet.ServletRequest的子类
Javax.servlet.jsp.PageContext

服务器总是要处理并发的情形,Servlet默认不是线程安全的,要自己编程处理

service()方法对于HttpServlet来说是通过doMethod()实现的
通常来说都是接受request,进行一些处理,产生response
对于HttpServlet,要求首先response一个头部,然后是具体内容

所有的request都实现了ServletRequest接口,它定义了参数,属性,协议信息和本地化信息
可以用request的getReader或getInputStream手动处理信息
对于HttpServlet则是一个HttpServletRequest对象,还包括URL,头部信息,查询串等等

所有的response都实现了ServletResponse接口
可以用getWriter或getOutputStream手动输出信息
HttpServletResponse还包括状态码和Cookie信息等


过滤器可以对requeset和response过滤,当然都包括头部和内容
过滤器本身并不产生response,它总是依赖于其他的Web资源
每个Web资源可以有0或多个filter,这个filterchain在Web资源被实例化的时候会被确定

Filter,FilterChain及FilterConfig接口定义在javax.servlet包中
通过实现Filter接口定义一个filter,doFilter()方法传递request,response,filterchain
filterchain的前面是n个filter,最后面是那个Web资源,会依次往后传递
不调用doFilter()也就是请求被拒绝

另外filter也有int()和destroy()
filter可以有初始参数,会通过FilterConfig对象传递到init()方法,整个filter中都可使用

filter要想修改response必须在response发送到客户端前截获它
可以传递一个标准输入流来包装response,然后那个Servlet会被阻塞
现在可以实现自己的response了
修改request同理

filter-mapping默认只会过滤request,也就是客户端发来的URL
自己内部的forward,include,errorpage不会过滤
/* 表示对所有URL都过滤
一个filter可以映射到多个web资源,一个web资源也可以映射多个filter

通过getRequestDispatcher("URL")来获得一个RequestDispatcher对象进行include或forward
可以从request调用,也可以从ServletContext调用
不同的是,request得到的可以不用/开头,ServletContext得到的要用/开头
他俩都是相对于/myWeb/根路径的
但如果URL不对的话,RequestDispatcher对象会是null,编程时要判断下

被include的web资源不能更改原来页面的头部

如果已经想用ServletOutputStream 或 PrintWriter对象输出了,再调用forward会报错
之前的那个request会作为forward之后的那个页面的request的一些属性存在

属性名为:javax.servlet.forward.[request-uri|context-path|servlet-path|path-info|query-string]

Enumeration a = req.getAttributeNames();
		while(a!=null&&a.hasMoreElements()){//得到转发前的request
			Object nextElement = a.nextElement();
			System.out.println(nextElement+"\t"+req.getAttribute((String)nextElement));
		}
		System.out.println("\n");

 


request.getSession()可以得到HttpSession对象,可以附加各种属性供会话调用
客户端无法通知一个session结束,只能设置几分钟后自动过期
调用session的某个方法会让这个时间重置,所以是闲置时间
不过服务器端可以调用invalidate()让会话失效,并删除session数据

要想使用session跟踪,要调用resoonse.encodeURL(URL)方法
当cookie不可用时,会自动将sessionId添加到url后面
通常这么写,response.encodeURL(request.getContextPath()+”/catalog”);


调用destroy()也不会立即终止Servlet执行,但等待一定时间后会即刻终止
用以下措施来保证destroy时所有services都已结束:
1.

publicclassShutdownExampleextendsHttpServlet{
privateintserviceCounter=0;//设置一个字段,表示正有几个线程调用该servlet
...
//有线程进入和退出时进行记录
protected synchronized void enteringServiceMethod(){
serviceCounter++;
}
protected synchronized void leavingServiceMethod(){
serviceCounter--;
}
protected synchronized int numServices(){
returnserviceCounter;
}
}
protected void service(HttpServletRequestreq,
HttpServletResponseresp)
throwsServletException,IOException{
enteringServiceMethod();
try{
super.service(req,resp);
}finally{
leavingServiceMethod();
}
}


2.还在调用service时,资源就被destroy了,那当然不行

public class ShutdownExample extends HttpServlet{
private boolean shuttingDown;
...
//同步方法来访问
protectedsynchronizedvoidsetShuttingDown(booleanflag){
shuttingDown=flag;
}
protectedsynchronizedbooleanisShuttingDown(){
returnshuttingDown;
}
}
public void destroy(){
//还有线程在使用的话,通知其要关闭了
if(numServices()>0){
setShuttingDown(true);
}
/*等待service关闭.*/
while(numServices()>0){
try{
Thread.sleep(interval);
}catch(InterruptedExceptione){
}
}
}


3.需要长时间执行的方法要经常判断该servlet是否要关闭了

public void doPost(...){
...
for(i=0;((i<lotsOfStuffToDo)&&
!isShuttingDown());i++){
try{
partOfLongRunningOperation(i);
}catch(InterruptedExceptione){
...
}
}
}













 

你可能感兴趣的:(servlet)