第6章 Java Servlet基础

目录

6.1 servlet的部署、创建与运行

6.1.1  源文件及字节码文件

1.Servlet类

2.字节码文件的保存 

6.1.2  编写部署文件web.xml

6.1.3  servlet的创建与运行

6.1.4  向servlet传递参数的值

6.2 servlet的工作原理

6.2.1  servlet 对象的生命周期

6.2.2  init方法

6.2.3  service方法

6.2.4  destroy方法

6.3 通过JSP页面访问servlet

6.4 共享变量

6.5 doGet和doPost方法

6.6 重定向与转发

6.7 使用session


6.1 servlet的部署、创建与运行

6.1.1  源文件及字节码文件

1Servlet

写一个创建servlet的类就是编写一个特殊类的子类,这个特殊的类就是javax.servlet.http包中的HttpServletHttpServlet实现了Servlet接口,实现了响应用户的方法(这些方法将在后续内容中讲述)。HttpServlet的子类被习惯地称作一个Servlet类,这样的类创建的对象习惯地被称作一个servlet

例子6_1中的Example6_1.javaExamle6_1类是一个简单的Servlet类。

package moon.sun;
import java.io.*;
import javax.servlet.*;
import javax.servlet.http.*;
public class Example6_1 extends HttpServlet{
    public void init(ServletConfig config) throws ServletException{
       super.init(config);
    }
    public void service(HttpServletRequest request,HttpServletResponse response) throws IOException{
      //设置响应的MIME类型:
       response.setContentType("text/html;charset=utf-8");
       PrintWriter out = response.getWriter();//获得向用户发送数据的输出流。
       out.println("<html><body bgcolor = #ffccff>");
       out.println("<h1>hello servlet,你好 servlet</h1>");
       out.println("</body></html>");
    } 
}

2.字节码文件的保存 

字节码文件按着类的包名对应的目录路径保存到Web服务目录中特定子目录中。包名moon.sun对应的目录路径是moon\sun,因此,保存到:

\ch6\WEB-INF\classes\moon\sun

可以事先将Java源文件,按照其包名,比如,包名为moon.sun,保存到

\ch6\WEB-INF\classes\moon\sun

然后用命令行进入moon\sun的父目录classes(不要进入moonsun目录)编译:

classes> javac -cp servlet-api.jar moon\sun\Example6_1.java

 

6.1.2  编写部署文件web.xml

部署文件web.xml

<?xml version="1.0" encoding="utf-8" ?>
<web-app>
    <servlet>
        <servlet-name>hello</servlet-name>
        <servlet-class>moon.sun.Example6_1</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/lookHello</url-pattern>
    </servlet-mapping>
</web-app>

对于webapps下的Web服务目录,如果修改并重新保存web.xml文件,Tomcat服务器就会立刻重新读取web.xml文件,因此,修改web.xml文件不必重新启动Tomcat服务器。但是,如果修改导致web.xml文件出现错误,Tomcat服务器就会关闭当前Web服务目录下的所有servlet的使用权限。所以必须保证web.xml文件正确无误,才能成功启动Tomcat服务器。但是,对于不是webapps下的Web服务目录,如果新建或修改了相应的web.xml文件,需要重新启动Tomcat服务器 

6.1.3  servlet的创建与运行

■用户就可以根据web.xml部署文件来请求Tomcat服务器创建并运行一个servlet

■如果Tomcat服务器没有名字为helloservlet,就会根据web.xml文件中servlet标记的子标记servlet-class指定的Servlet类创建一个名字为helloservlet。因此,如果名字是helloservlet被创建之后,又修改Java源文件、编译得到新的Servlet类,并希望Tomcat服务器用新的Servlet类创建servlet,那么就要重新启动Tomcat服务器。根据6.1.2中的web.xml文件,用户需在浏览器输入:

http://127.0.0.1:8080/ch6/lookHello

请求Tomcat服务器运行名字是helloservlet

6.1.4  servlet传递参数的值

在请求一个servlet时,可以在请求的url-pattern中额外加入参数及其值,格式是:

url-pattern?参数1=&参数2=参数n=

那么被请求的servlet就可以使用request对象获取参数的值,例如:

request.getParameter(参数n)

例子6_2中,用户请求servlet绘制一个椭圆,请求时将椭圆的宽(横轴长)和高(纵轴长)通过参数widthheight传递给servlet

package moon.sun;
import java.awt.image.BufferedImage;
import java.awt.*;
import java.awt.geom.*; 
import javax.imageio.ImageIO;
import java.io.*;
import javax.servlet.*;
import javax.servlet.http.*;
public class Example6_2 extends HttpServlet{
    public void init(ServletConfig config) throws ServletException{
       super.init(config);
    }
    public void service(HttpServletRequest request,HttpServletResponse response) throws IOException{
        request.setCharacterEncoding("utf-8");
        double width  = Double.parseDouble(request.getParameter("width"));
        double height = Double.parseDouble(request.getParameter("height"));
        response.setContentType("image/jpeg");
        Ellipse2D ellipse=new Ellipse2D.Double(400-width/2,300-height/2,width,height);
        BufferedImage image = getImage(ellipse);
        try {
            OutputStream outClient= response.getOutputStream();//指向用户端的输出流。
            boolean boo =ImageIO.write(image,"jpeg",outClient);
        }
        catch(Exception exp){}
    } 
    BufferedImage getImage(Shape shape){ //得到图形的图像
        int width=800, height=600;
        BufferedImage image = 
        new BufferedImage(width,height,BufferedImage.TYPE_INT_RGB);
        Graphics g = image.getGraphics();
        g.fillRect(0, 0, width, height);
        Graphics2D g_2d=(Graphics2D)g; 
        g_2d.setColor(Color.blue);
        g_2d.fill(shape); 
        return image;
    }
}

 

 

6.2 servlet的工作原理

servletTomcat服务器负责管理,Tomcat服务器通过读取web.xml,然后创建并运行servlet

6.2.1  servlet 对象的生命周期

servletTomcat服务器负责创建并完成初始化工作。当多个用户请求一个servlet时,服务器为每个用户启动一个线程。

一个servlet的生命周期主要有下列三个过程组成:

(1) servlet第一次被请求加载时,服务器创建servletservlet调用init方法完成必要的初始化工作。

(2) 新诞生的servlet再调用service方法响应用户的请求。

(3) 当服务器关闭时,调用destroy方法销毁servlet

init方法只被调用一次。当后续的用户请求servlet服务时,Tomcat服务器将启动一个新的线程,在该线程中,servlet调用service方法。也就是说,每个用户的每次请求都导致service方法被调用执行,其执行过程分别运行在不同的线程中。

6.2.2  init方法

public void init(ServletConfig  config) throws ServletException

servlet第一次被请求加载时,服务器创建一个servlet,这个对象调用init方法完成必要的初始化工作。该方法在执行时,服务器会把一个SevletConfig类型的对象传递给init()方法,这个对象就被保存在servlet中,直到servlet被销毁。

6.2.3  service方法

public void service(HttpServletRequest request  HttpServletResponse  response)throw ServletException,IOException

Tomcat服务器将两个参数传递给该方法。和init方法不同的是,init方法只被调用一次,而service方法可能被多次的调用。当后续的用户请求该servlet时,Tomcat服务器将启动一个新的线程,在该线程中servlet调用service方法响应用户的请求,调用过程运行在不同的线程中,互不干扰。因此,不同线程的service方法中的局部变量互不干扰,一个线程改变了自己的service方法中局部变量的值不会影响其他线程的service方法中的局部变量。

6.2.4  destroy方法

public destroy()

Tomcat服务器终止服务时,destroy()方法会被执行,销毁servlet.子类可直接继承这个方法,一般不需要重写。

6.3 通过JSP页面访问servlet

按着部署文件web.xml给出的servleturl-pattern,用户除了可以在浏览器输入url-pattern请求运行一个servlet外,也可以通过JSP页面来请求一个servlet

需要特别注意的是,如果web.xml文件中给出的servleturl-pattern/lookHello,那么JSP页面请求servlet时,必须要写lookHello不可以写成/lookHello

■通过表单向servlet提交数据

如果web.xml文件中给出的servleturl-pattern/computeBill,那么form表单中action给出的值就是computeBill

<form  action= "computeBill "  >

</form>

例子6_3中,JSP页面example6_3.jsp通过form表单请求名字是computeBillservlet,向所请求的computeBill提交一个账单明细,在请求computeBillurl-pattern中额外加入一个参数discount,其值代表优惠额度,例如discount值是6时,表示打6折。

<%@ page contentType="text/html" %>
<%@ page pageEncoding = "utf-8" %>
<style>
   #textStyle{
      font-family:宋体;font-size:36;color:blue 
   }
</style>
<HTML><body bgcolor=#ffccff>
<form action="computeBill?discount=6" id= textStyle method=post>
  输入账单:<br>
<textArea name='billMess' id =textStyle rows=5 cols=30 >
 剁椒鱼头:62.9元,烤鸭:199元,红焖大虾:289.9元。
</textArea>
<br><input type=submit id=textStyle value="提交">
</form>
</body></HTML>

 

 

6.4 共享变量

Servlet类是HttpServlet的一个子类,在编写子类时就可以声明某些成员变量,那么,请求servlet的用户将共享该servlet的成员变量

service方法可能被多次的调用。也就是说,当后续的用户请求该servlet时,Tomcat服务器将启动一个新的线程,在该线程中servlet调用service方法响应用户的请求,即每个用户的请求都导致service方法被调用执行,调用过程运行在不同的线程中,互不干扰。因此,不同线程的service方法中的局部变量互不干扰,一个线程改变了自己的service方法中局部变量的值不会影响其他线程的service方法中的局部变量。

6.5 doGetdoPost方法

HttpServlet类除了initservicedestroy方法外,该类还有两个很重要的方法:doGetdoPost,用来处理用户的请求并作出响应。

■实际上HttpServlet类所给出的service方法的功能是检查HTTP请求类型(getpost),并在service方法中根据用户的请求方式,在service方法中对应地再调用doGetdoPost方法。

■因此,在编写的Servlet类(HttpServlet类的一个子类)时,也可以不重写service方法来响应用户,直接继承service方法即可。

如果不重写service方法就需要在Servlet类中重写doPostdoGet方法来响应用户的请求。如果不论用户请求类型是post还是getTomcat服务器的处理过程完全相同,那么可以只在doPost方法中编写处理过程,而在doGet方法中再调用doPost方法即可,或只在doGet方法中编写处理过程,而在doPost方法中再调用doGet方法。如果根据请求的类型进行不同的处理,就要在两个方法中编写不同的处理过程(这一点比service方法更为灵活)

例子6_6example6_5.jsp页面有两个form表单,每个form表单都请求名字为sumORproductservlet,并提交一串数字给sumORproduct。一个form表单的请求方式是post,另一个form表单的请求方式是get。当form表单的请求方式是post时,sumORproduct计算各数字的代数和,当form表单的请求方式是get时,sumORproduct计算各数字的乘积。

<%@ page contentType="text/html" %>
<%@ page pageEncoding = "utf-8" %>
<style>
   #textStyle{
      font-family:宋体;font-size:36;color:blue 
   }
</style>
<HTML><body bgcolor=#ffccff>
<form action="sumORproduct" id =textStyle method=post>
  输入数字,用逗号分隔提交给servlet(post方式):
  <br><input type=text id =textStyle name="number"/>
  <input type=submit id =textStyle value="提交">
</form>
<form action="sumORproduct" id =textStyle method=get>
   输入数字,用逗号分隔提交给servlet(get方式):
  <br><input type=text id =textStyle name="number"/>
  <input type=submit id =textStyle value="提交"/>
</form>
</body></HTML>

 

 

6.6 重定向与转发

重定向的功能是将用户从当前页面或servlet定向到另一个JSP页面servlet转发的功能是将用户对当前JSP页面或servlet请求转发给另一个JSP页面servlet。本节学习在Servlet类中使用HttpServletResponse类的sendRedirect重定向方法,以及RequestDispatcher类的forward转发方法,并指出二者的区别。

1.sendRedirect方法

重定向方法void sendRedirect(String location)

将用户重新定向到另一个JSP页面或servlet。重定向方法仅仅是将用户从当前页面或servlet定向到另一个JSP页面或servlet,但不能将用户对当前页面或servlet的请求(HttpServletRequest对象)转发给所定向的资源。即重定向的目标页面或servlet无法使用request获取用户提交的数据。

执行sendRedirect方法(重定向,也见4.2.3)时,Tomcat服务器还是要把当前的servlet代码执行完毕后才实施重定向(跳转)操作,但Tomcat服务器不再给用户看当前servlet代码的执行效果。如果在执行sendRedirect(URL url)方法后,servlet紧接着执行了return返回语句,那么Tomcat服务器会立刻结束当前servlet的执行。

2.forward方法

  RequestDispatcher dispatcher =

  request.getRequestDispatcher(JSP页面的URLservleturl-pattern);

例如:

RequestDispatcher dispatcher = request.getRequestDispatcher("target.jsp");

转发。在步骤(1)中获取的RequestDispatcher对象调用

void forward(ServletRequest request,ServletResponse response)             throws ServletException,ava.io.IOException

方法可以将用户对当前JSP页面或servlet的请求转发给RequestDispatcher对象所指定的JSP页面或servlet,例如:

dispatcher.forward (request,response);

把用户对当前JSP页面或servlet的请求转变为对转发到的JSP页面或servlet的请求

3.二者的区别

■转发(forwar)和重定向方法(sendRedirect)不同的是,用户可以看到转发到的JSP页面或servlet的运行效果,但是,在浏览器的地址栏中不能看到forward方法转发到的JSP页面的地址或servlet的地址,用户在浏览器的地址栏中所看到的仍然是当前JSP页面的URLservleturl-pattern。如果此时刷新浏览器,那么请求将是当前的JSP页面或servlet所转发到的标页面或servlet可以使用request获取用户提交的数据。而重定向的目标页面或servlet无法使用request获取用户提交的数据。

■另外,当servlet中执行forward方法实施转发操作时,Tomcat会立刻结速当前servlet的执行。而servlet中执行sendRedirect方法(重定向,也见4.2.3)时,Tomcat服务器还是要把当前的servlet代码执行完毕后才实施重定向(跳转)操作,但Tomcat服务器不再给用户看当前servlet代码的执行效果。如果在执行sendRedirect(URL url)方法后,servlet紧接着执行了return返回语句,那么Tomcat服务器会立刻结束当前servlet的执行。

6.7 使用session

HTTP通信协议是用户与服务器之间一种请求与响应(request/response)的通信协议,属于无状态协议。所谓无状态是指,当用户(浏览器)发送请求给服务器,Tomcat服务器作出响应后,如果同一个用户再发送请求给Tomcat服务器时,Tomcat服务器并不知道就是刚才的那个用户。简单地说,Tomcat服务器不会记录用户的信息。

用户在访问一个Web服务目录期间,Tomcat服务器为该用户分配一个session对象(称为用户的会话),Tomcat服务器可以在各个页面以及servlet中使用这个session记录用户的有关信息,而且Tomcat服务器保证不同用户的session对象互不相同。有关session对象的原理、常用方法可参见第4章的4.3

例子6_8是一个猜字母游戏

用户访问 刷新 example6_8.jsp 页面时,随机分配给用户一个英文字母( 不区分大小写),并将这个字母存在用户的 session
用户 链接到 example6_8_input.jsp 页面输入自己的猜测,并将该猜测提交给一个名字为 guess servlet ,该 servlet 负责处理用户的猜测
具体 处理方式是:如果用户猜小了,将“猜小了”存放到用户的 session 中,然后将用户重新定向到 example6_8_input.jsp 。如果猜大了,将“猜大了”存放到用户的 session 中,然后将用户重新定向到 example6_8_input.jsp
如果 用户猜成功了,将“猜对了”存放到用户的 session 中,然后将用户重新定向到 example5_8_input.jsp
<%@ page contentType="text/html" %>
<%@ page pageEncoding = "utf-8" %>
<HTML><body bgcolor=#ffccff >
<% session.setAttribute("message","请您猜字母");
   session.setAttribute("count","0");
   char a[]=new char[26];
   int m = 0;
   for(char c='a';c<='z';c++) {
      a[m] = c;
      m++;
   }
   int randomIndex=(int)(Math.random()*a.length);      
   char ch=a[randomIndex];  //获取一个英文字母
   session.setAttribute("savedLetter",new Character(ch));//将该字母放入session中
%>
<h3>访问或刷新该页面可以随机得到一个英文字母。
<br>单击超链接去猜出这个字母:<br>
<a href="example6_8_input.jsp">去猜字母</a>
</h3></body></HTML>

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值