Servlet的引入
之前我们知道安装tomcat后,通过修改配置信息等,启动tomcat,在浏览器中输入你本机网址+虚拟路径+真实文件路径下的文件就可以访问网页了,但这都是根据你真实路径下存在这么一个文件,都是静态的页面信息,如果我想动态的访问页面,比如说,我在页面提交表单数据,浏览器就应该要给我返回一些动态的相关信息,这时就需要用到Servlet技术了
Servlet的概述
Servlet----Server Applet (java servlet的简称),称为小服务程序或服务连接器,用java编写的服务端程序,具有独立于平台和协议的特性,主要功能在于交互式地浏览和生成数据,生成动态Web内容。
总体来说,就是用来处理请求和发送响应的过程,Servlet是为了解决实现动态页面而衍生的东西
B/S模式:
Servlet和tomcat的关系
tomcat是Web应用服务器,是一个Servlet/jsp容器,tomcat作为servlet容器,负责处理客户请求,把客户请求传递给servlet,并将servlet的响应发送回给客户。而servlet是一种运行在支持java语言的服务器上的组件,servlet最常见的用途就是可以拓展java web服务器功能,能提供非常安全的,可移植的,易于使用的CGI替代品
从http协议中的请求和响应可以得知,浏览器发出的请求是一个请求文本,而浏览器接收到的也应该是一个响应文本。但是在上面这个图中,并不知道是如何转变的,只知道浏览器发送过来的请求也就是request,我们响应回去的就用response。忽略了其中的细节,现在就来探究一下。
1、Tomcat将http请求文本接收并解析,然后封装成HttpServletRequest类型的request对象,所有的HTTP数据头可以通过request对象调用对应的方法查询到。
2、Tomcat同时会要响应的信息封装为HttpServletResponse类型的response对象,通过设置response属性就可以控制要输出到浏览器的内容,然后将response交给tomcat,tomcat就会将其变成响应文本的格式发送给浏览器
Java Servlet API 是Servlet容器(tomcat)和servlet之间的接口,它定义了serlvet的各种方法,还定义了Servlet容器传送给Servlet的对象类,其中最重要的就是ServletRequest和ServletResponse。所以说我们在编写servlet时,需要实现Servlet接口,按照其规范进行操作。
Servlet的使用
如果要开发一个web资源,如开发一个java程序向浏览器输出数据,只需要实现两部操作即可:
1、编写一个java类实现Servlet接口
2、把开发好的java类部署到web服务器
Servlet的运行过程
Servlet程序是由WEB服务器调用,web服务器收到客户端的Servlet访问请求后:
1、Web服务器首先检查是否已经装载并创建了该Servlet的实例对象。如果是,则直接执行第4步,否则,执行第2步。
2、装载并创建该Servlet的一个实例对象。
3、调用Servlet实例对象的init()方法。
4、创建一个用于封装HTTP请求消息的HttpServletRequest对象和一个代表HTTP响应消息的HttpServletResponse对象,然后调用Servlet的service()方法并将请求和响应对象作为参数传递进去。
5、Web应用程序被停止或重新启动之前,Servlet引擎将卸载Servlet,并在卸载之前调用Servlet的destroy()方法。
Servlet调用图
在STS(Spring Tool Suit,eclipse的一个插件)中开发Servlet
1、在STS中新建一个动态项目MvcProject
打上勾,目的是生成项目的配置文件web.xml
2、为工作空间添加tomcat服务
让server显示出来
添加服务
3、将项目部署到tomcat服务器中
4、添加环境
MvcProject:表示web工程名称,该工程部署时,在webapps目录下就会有一个MvcProject的web应用
src:表示java程序的开发目录,该目录下编写的所有j程序在部署时,会自动部署到MvcProject/web-inf/classes目录下
WebContent:表示对应于web应用的根目录,该目录下的所有子目录和子文件在部署时,会原封不动的发布到web应用目录下
1、Servlet接口实现类
Servlet接口SUN公司定义了两个默认实现类,分别为:GenericServlet、HttpServlet。
HttpServlet指能够处理HTTP请求的servlet,它在原有Servlet接口上添加了一些与HTTP协议处理方法,它比Servlet接口的功能更为强大。因此在编写Servlet时,通常继承这个类,而避免直接去实现Servlet接口。
HttpServlet在实现Servlet接口时,覆写了service方法,该方法体内的代码会自动判断用户的请求方式,如为GET请求,则调用HttpServlet的doGet方法,如为Post请求,则调用doPost方法。因此,开发人员在编写Servlet时,通常只需要覆写doGet或doPost方法,而不要去覆写service方法。
HttpServlet是一个抽象类,该抽象类中没有抽象方法,是一个模板设计模式的类,模板方法是service(),我们来看看它的源码
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String method = req.getMethod();
long lastModified;
if (method.equals("GET")) {
lastModified = this.getLastModified(req);
if (lastModified == -1L) {
this.doGet(req, resp);
} else {
long ifModifiedSince;
try {
ifModifiedSince = req.getDateHeader("If-Modified-Since");
} catch (IllegalArgumentException var9) {
ifModifiedSince = -1L;
}
if (ifModifiedSince < lastModified / 1000L * 1000L) {
this.maybeSetLastModified(resp, lastModified);
this.doGet(req, resp);
} else {
resp.setStatus(304);
}
}
} else if (method.equals("HEAD")) {
lastModified = this.getLastModified(req);
this.maybeSetLastModified(resp, lastModified);
this.doHead(req, resp);
} else if (method.equals("POST")) {
this.doPost(req, resp);
} else if (method.equals("PUT")) {
this.doPut(req, resp);
} else if (method.equals("DELETE")) {
this.doDelete(req, resp);
} else if (method.equals("OPTIONS")) {
this.doOptions(req, resp);
} else if (method.equals("TRACE")) {
this.doTrace(req, resp);
} else {
String errMsg = lStrings.getString("http.method_not_implemented");
Object[] errArgs = new Object[]{method};
errMsg = MessageFormat.format(errMsg, errArgs);
resp.sendError(501, errMsg);
}
}
编写一个类,继承HttpServlet类
@SuppressWarnings("serial")
public class EmpServlet extends HttpServlet{
@Override
protected void service(HttpServletRequest arg0, HttpServletResponse arg1) throws ServletException, IOException {
System.out.println("开启服务时调用");
}
}
此时Servlet开发完成,但是这个servlet负责处理的路径是什么呢,或者说他负责处理的路径是谁分配的,是容器tomcat去分配。分配之前需对配置信息进行修改,在web.xml中进行配置,在webContent/WEB-INF下
由于客户端是通过URL地址访问web服务器中的资源,所以Servlet程序若想被外界访问,必须把servlet程序映射到一个URL地址上,这个工作在web.xml文件中使用元素和元素完成。
元素用于注册Servlet,它包含有两个主要的子元素:和,分别用于设置Servlet的注册名称和Servlet的完整类名。
一个元素用于映射一个已注册的Servlet的一个对外访问路径,它包含有两个子元素:和,分别用于指定Servlet的注册名称和Servlet的对外访问路径。
配置添加以下内容
<servlet>
<servlet-name>empServlet</servlet-name>
<servlet-class>com.zhouym.mvcpro.controller.EmpServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>empServlet</servlet-name>
<url-pattern>/emp/*</url-pattern>
</servlet-mapping>
同一个Servlet可以被映射到多个URL上,即多个元素的子元素的设置值可以是同一个Servlet的注册名。
<servlet>
<servlet-name>empServlet</servlet-name>
<servlet-class>com.zhouym.mvcpro.controller.EmpServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>empServlet</servlet-name>
<url-pattern>/emp/*</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>empServlet</servlet-name>
<url-pattern>/emp01/index.html</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>empServlet</servlet-name>
<url-pattern>/zhouym.jsp</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>empServlet</servlet-name>
<url-pattern>/emp03</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>empServlet</servlet-name>
<url-pattern>/emp04</url-pattern>
</servlet-mapping>
通过上面的配置,当我们想访问名称是empServlet的Servlet,可以使用如下的几个地址去访问:
/*通配符代表所有,emp下随便输入字符就可以访问
http://localhost:8080/MvcProject/emp/asjdjas
http://localhost:8080/MvcProject/emp01/index.html
http://localhost:8080/MvcProject/zhouym.jsp
等等
在启动项目之前关闭tomcat,不能开启多个tomcat窗口,否则就会被提示端口被占用,出现如下表示启动成功
在浏览器中访问:
访问路径:主机名(域名/IP)+项目名称+servlet的路径信息
eg:http://localhost:8080/MvcProject/emp/wdjasdblmm