第一部分
(今天的总结不用写代码)
Http协议介绍
HTTP协议是Hyper Text Transfer Protocol(超文本传输协议)的缩写,是用于从万维网(WWW:World Wide Web )服务器传输超文本到本地浏览器的传送协议。总结来说就是,> 双方在交互、通讯的时候, 遵守的一种规范、规则。其实就是规定了客户端在访问服务器端的时候,要带上哪些东西, 服务器端返回数据的时候,也要带上什么东西。
Http协议的版本
1.0
请求数据,服务器返回后, 将会断开连接
1.1
请求数据,服务器返回后, 连接还会保持着。 除非服务器 | 客户端 关掉。 有一定的时间限制,如果都空着这个连接,那么后面会自己断掉。
举例子加深我们的认识:
需要的实验环境
(1)httpwatch抓包插件
(2)浏览器打开一个在线登录界面(例如在线邮箱登录)
步骤:
安装抓包软件(默认安装就可以),打开IE浏览器,在状态栏即可找到安装的抓包插件;
在线登录界面输入账号密码后,点击抓包插件里面的record,即可完成抓包动作。
下面是我利用tomcat的例子,和上面的例子类似,我以现在的例子作详细解释:
###Http请求数据解释
请求的数据里面包含三个部分内容 : 请求行 、 请求头 、请求体
-
请求行
POST /examples/servlets/servlet/RequestParamExample HTTP/1.1 POST : 请求方式 ,以post去提交数据 /examples/servlets/servlet/RequestParamExample 请求的地址路径 , 就是要访问哪个地方。 HTTP/1.1 协议版本
-
请求头
Accept: application/x-ms-application, image/jpeg, application/xaml+xml, image/gif, image/pjpeg, application/x-ms-xbap, */* Referer: http://localhost:8080/examples/servlets/servlet/RequestParamExample Accept-Language: zh-CN User-Agent: Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.1; WOW64; Trident/4.0; SLCC2; .NET CLR 2.0.50727; .NET CLR 3.5.30729; .NET CLR 3.0.30729; Media Center PC 6.0; .NET4.0C; .NET4.0E) Content-Type: application/x-www-form-urlencoded Accept-Encoding: gzip, deflate Host: localhost:8080 Content-Length: 31 Connection: Keep-Alive Cache-Control: no-cache Accept: 客户端向服务器端表示,我能支持什么类型的数据。 Referer : 真正请求的地址路径,全路径 Accept-Language: 支持语言格式 User-Agent: 用户代理 向服务器表明,当前来访的客户端信息。 Content-Type: 提交的数据类型。经过urlencoding编码的form表单的数据 Accept-Encoding: gzip, deflate : 压缩算法 。 Host : 主机地址 Content-Length: 数据长度 Connection : Keep-Alive 保持连接 Cache-Control : 对缓存的操作
-
请求体
浏览器真正发送给服务器的数据
发送的数据呈现的是key=value ,如果存在多个数据,那么使用 &
firstname=zhang&lastname=sansan
###Http响应数据解析
请求的数据里面包含三个部分内容 : 响应行 、 响应头 、响应体
HTTP/1.1 200 OK
Server: Apache-Coyote/1.1
Content-Type: text/html;charset=ISO-8859-1
Content-Length: 673
Date: Fri, 17 Feb 2017 02:53:02 GMT
...这里还有很多数据...
-
响应行
HTTP/1.1 200 OK 协议版本 状态码 咱们这次交互到底是什么样结果的一个code. 200 : 成功,正常处理,得到数据。 403 : for bidden 拒绝 404 : Not Found 500 : 服务器异常 OK 对应前面的状态码
-
响应头
Server: 服务器是哪一种类型。 Tomcat Content-Type : 服务器返回给客户端你的内容类型 Content-Length : 返回的数据长度 Date : 通讯的日期,响应的时间
第二部分:
Servlet简介
其实就是一个java程序,运行在我们的web服务器上,用于接收和响应 客户端的http请求。 更多的是配合动态资源来做。 当然静态资源也需要使用到servlet,只不过是Tomcat里面已经定义好了一个 DefaultServlet
现在举例子体验基本的servlet工作原理:
1. 得写一个Web工程 , 要有一个服务器。
2. 测试运行Web工程
1. 新建一个类, 实现Servlet接口
2. 配置Servlet , 用意: 告诉服务器,我们的应用有这么些个servlet。
在webContent/WEB-INF/web.xml里面写上以下内容。
<!-- 向tomcat报告, 我这个应用里面有这个servlet, 名字叫做HelloServlet , 具体的路径是com.itheima.servlet.HelloServlet -->
<servlet>
<servlet-name>HelloServlet</servlet-name>
<servlet-class>com.itheima.servlet.HelloServlet</servlet-class>
</servlet>
<!-- 注册servlet的映射。 servletName : 找到上面注册的具体servlet, url-pattern: 在地址栏上的path 一定要以/打头 -->
<servlet-mapping>
<servlet-name>HelloServlet</servlet-name>
<url-pattern>/a</url-pattern>
</servlet-mapping>
3. 在地址栏上输入 http://localhost:8080/项目名称/a
Servlet的执行过程如下图:
Servlet的生命周期:
掌握客户端访问,初始化多少次;
Servlet的请求方法是否每访问一次就会执行一次;
Servlet是什么时候被销毁(就是生命周期的结束)
package com.itheima.servlet;
import java.io.IOException;
import javax.servlet.Servlet;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServlet;
/**
* 该servlet 用于演示servlet的生命周期调用时机
*/
public class HelloServlet03 implements Servlet{
//1、
//在创建该servlet的实例时,就执行该方法。
//一个servlet只会初始化一次, init方法只会执行一次
//默认情况下是 : 初次访问该servlet,才会创建实例。
@Override
public void init(ServletConfig config) throws ServletException {
System.out.println("HelloServlet03 初始化...");
}
//2.
/*
* 只要客户端来了一个请求,那么就执行这个方法了。
* 该方法可以被执行很多次。 一次请求,对应一次service方法的调用
*/
@Override
public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {
System.out.println("helloservlet03 service方法调用了");
}
/**
* servlet销毁的时候,就会执行该方法
*
* 1. 该项目从tomcat的里面移除。
* 2. 正常关闭tomcat就会执行 shutdown.bat
*/
@Override
public void destroy() {
System.out.println("helloservlet03 destroy方法调用了");
}
@Override
public ServletConfig getServletConfig() {
return null;
}
@Override
public String getServletInfo() {
// TODO Auto-generated method stub
return null;
}
}
Servlet只会有一次初始,每访问一次SERVLET的service就会调用一次方法
Servlet的销毁:从tomcat中移除或者从关闭tomcat
让Servlet创建实例的时机 提前
-
默认情况下,只有在初次访问servlet的时候,才会执行init方法。 有的时候,我们可能需要在这个方法里面执行一些初始化工作,甚至是做一些比较耗时的逻辑。
-
那么这个时候,初次访问,可能会在init方法中逗留太久的时间。 那么有没有方法可以让这个初始化的时机提前一点。
-
在配置的时候, 使用load-on-startup元素来指定, 给定的数字越小,启动的时机就越早。 一般不写负数, 从2开始即可。
<servlet> <servlet-name>HelloServlet04</servlet-name> <servlet-class>com.itheima.servlet.HelloServlet04</servlet-class> <load-on-startup>2</load-on-startup> </servlet>
读取web配置参数
如何在类中读取配置中的信息,下面例子(源码自行云盘下载):
package com.itheima.servlet;
import java.io.IOException;
import java.util.Enumeration;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class HelloServletConfig extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//ServletConfig 可以获取servlet在配置的一些信息
//1. 得到servlet配置对象 专门用于在配置servlet的信息
ServletConfig config = getServletConfig();
//获取到的是配置servlet里面servlet-name 的文本内容
String servletName = config.getServletName();
System.out.println("servletName="+servletName);
//2、。 可以获取具体的某一个参数。
String address = config.getInitParameter("address3333");
System.out.println("address="+address);
System.out.println("----------------------------------");
//3.获取所有的参数名称
Enumeration<String> names = config.getInitParameterNames();
//遍历取出所有的参数名称
while (names.hasMoreElements()) {
String key = (String) names.nextElement();
String value = config.getInitParameter(key);
System.out.println("key==="+key + " value="+value);
}
}
//来了post请求,就让它去执行doGet方法
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
为什么需要有这个ServletConfig
-
未来我们自己开发的一些应用,使用到了一些技术,或者一些代码,我们不会。 但是有人写出来了。它的代码放置在了自己的servlet类里面。 也就是在加载配置文件时候,可以把额外的资源文件加载到服务器。
-
刚好这个servlet 里面需要一个数字或者叫做变量值。 但是这个值不能是固定了。 所以要求使用到这个servlet的公司,在注册servlet的时候,必须要在web.xml里面,声明init-params
在开发当中比较少用。