此博客用于个人学习,来源于网上,对知识点进行一个整理。
一、Http协议:
1.Http协议的定义:
-
什么是协议
双方在交互、通讯的时候,遵守的一种规范、规则。
-
什么是http协议
针对网络上的客户端与服务器端在执行http请求的时候,遵守的一种规范。其实就是规定了客户端在访问服务器端的时候,要带上哪些东西,服务器端返回数据的时候,也要带上什么东西。
-
版本
1.0 请求数据,服务器返回后,将会断开连接
1.1 请求数据,服务器返回后,连接还会保持着。除非服务器 | 客户端 关掉。有一定的时间限制,如果都 空着这个连接,那么后面会自己断掉。
2.演示客户端如何与服务器端进行通讯:
在地址栏中键入网络地址回车或者是平常注册的时候,点击了注册按钮 ,浏览器都能显示出来一些东西。那么背地里到底浏览器和服务器是怎么通讯,它们都传输了哪些数据。
- 安装抓包工具 HttpWatch (IE插件)
- 打开 tomcat. 输入 localhost:8080 打开首页
- 在首页上找到 Example 字样
- 选择 servlet 例子 —> Request Parameter
接着点击 Request Parameters 的 Execute 超链接
执行 tomcat 的例子,然后查看浏览器和 tomcat 服务器的对接细节(由于此电脑的 IE 浏览器对接错误,使用其他图片进行解析)
3.Http请求数据解释:
请求的数据里面包含三个部分内容 : 请求行 、 请求头 、请求体。
1)请求行:
POST /examples/servlets/servlet/RequestParamExample HTTP/1.1
POST: 请求方式 ,以 post 去提交数据。
examples/servlets/servlet/RequestParamExample: 请求的地址路径,即访问的地方。
HTTP/1.1: 协议版本。
2)请求头:
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: 用户代理,即客户端信息,例如Android和PC虽然都能登录同一个网站,但是信息不同。
Content-Type: 提交的数据类型,-form-urlencoded为经过urlencoding编码的form表单的数据。
Accept-Encoding: gzip, deflate,允许的压缩算法。
Host: 主机的地址。
Content-Length: 数据的长度。
Connection: Keep-Alive,保持连接。
Cache-Control: 对缓存的操作。
3)请求体:
浏览器真正发送给服务器的数据
firstname=zhang&lastname=sansan
发送的数据呈现的是 key=value ,如果存在多个数据,那么使用&,方便服务器对数据进行解析。
4.Http响应数据解析:
请求的数据里面包含三个部分内容 : 响应行 、 响应头 、响应体
1)响应行:
HTTP/1.1 200 OK
HTTP/1.1: 协议的版本号。
200: 这次交互结果的一个 code,常见的 code 有:
-
200 : 成功,正常处理,得到数据。
-
403:for bidden,拒绝连接。
-
404:not found。
-
500:服务器异常。
-
OK: 对应前面的状态码。
2)响应头:
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
Server: 服务器类型,Apache-Coyote/1.1 为 tomcat。
Content-Type: 服务器返回给客户端的内容类型。
Content-Length: 返回的数据长度。
Date: 通讯的日期,响应的时间。
5.POST 与 GET 的请求区别:
1)从两者返回的请求数据解析来看:
2)从性能来看:
-
POST
1,数据是以流的方式写过去,不会在地址栏上面显示,能够保护客户端的信息。现在一般提交数据到服务器使用的都是 POST。
2,以流的方式写数据,所以数据没有大小限制。
-
GET
1,会在地址栏后面拼接数据,所以有安全隐患。 一般从服务器获取数据,并且客户端也不用提交上面数据的时候,可以使用 GET。
2,能够带的数据有限, 1kb大小
二、Servlet:
1.Web 资源:
在 http 协议当中,规定了请求和响应双方, 客户端和服务器端。与 web 相关的资源。
有两种:
-
静态资源
html,css,js
-
动态资源
servlet,jsp
2.Servlet:
其实就是一个 java 程序,运行在我们的 web 服务器上,用于接收和响应客户端的http请求。 更多的是配合动态资源来做。 当然静态资源也需要使用到 servlet,只不过是 Tomcat 里面已经定义好了一个 DefaultServlet。
3.HelloServlet:
- 得写一个 Web 工程 , 要有一个服务器。
- 测试运行 Web 工程
- 新建一个类, 实现 Servlet 接口
- 配置 Servlet , 用意: 告诉服务器,我们的应用有这么些个 servlet。
在 webContent/WEB-INF/web.xml 里面写上以下内容。
<!-- 向tomcat报告,该应用有一个Servlet,名字叫HelloServlet,具体路径为myServlet.HelloServlet -->
<servlet>
<servlet-name>HelloServlet</servlet-name>
<servlet-class>myServlet.HelloServlet</servlet-class>
</servlet>
<!-- 注册的Servlet的映射,servlet-name用于找到上面的servlet名称,url-pattern为地址栏上的path -->
<servlet-mapping>
<servlet-name>HelloServlet</servlet-name>
<url-pattern>/a</url-pattern>
</servlet-mapping>
先在项目旁边开启 Tomcat,新建一个类,继承接口 Servlet,重新实现抽象方法 service,然后在浏览器地址栏输入:http://localhost:8080/HelloWeb/a,然后在 java 的控制台可以看到 service 实现的内容。
4.Servlet 执行过程:
5.Servlet 的通用写法:
类的关系:Servlet (接口)——>GenericServlet——>HttpServlet (用于处理http的请求)
由于 Servlet 的方法并不需要都实现,所有书写一个类,继承 HttpServlet 接口
public class HelloServlet02 extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// TODO 自动生成的方法存根
super.doGet(req, resp);
System.out.println("执行get操作");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// TODO 自动生成的方法存根
super.doPost(req, resp);
System.out.println("执行post操作");
}
}
6.Servlet的生命周期:
-
生命周期:从创建到销毁的那一段时间
-
生命周期方法
从创建到销毁,所调用的方法,doGet 和 doPost不算生命周期方法,所谓的生命周期方法是指,从对象的创建到销毁一定会执行的方法, 但是这两个方法,不一定会执行。
-
三个方法:init,service,destroy
/*
* 在创建该servlet的实例时,就执行该方法。
* 一个servlet只会初始化一次,init方法只会执行一次 默认情况下是 :
* 初次访问该servlet,才会创建实例。
*/
@Override
public void init(ServletConfig config) throws ServletException {
// TODO 自动生成的方法存根
System.out.println("HelloServlet03 init执行");
}
/*
* 只要客户端来了一个请求,那么就执行这个方法了。
* 该方法可以被执行很多次。一次请求,对应一次service方法的调用。
*/
@Override
public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {
// TODO 自动生成的方法存根
System.out.println("HelloServlet03 service执行");
}
/*
* servlet销毁的时候,就会执行该方法
* 1.该项目从tomcat的里面移除。
* 2.正常关闭tomcat就会执行 shutdown.bat。
*/
@Override
public void destroy() {
// TODO 自动生成的方法存根
System.out.println("HelloServlet03 destroy执行");
}
7.让Servlet创建实例的时机提前:
-
默认情况下,只有在初次访问 servlet 的时候,才会执行 init 方法。有的时候,我们可能需要在这个方法里面执行一些初始化工作,甚至是做一些比较耗时的逻辑。
-
那么这个时候,初次访问,可能会在 init 方法中逗留太久的时间。那么有没有方法可以让这个初始化的时机提前一点。
-
在配置的时候,使用 load-on-startup 元素来指定,给定的数字越小,启动的时机就越早。一般不写负数,从2开始即可。
<servlet>
<servlet-name>HelloServlet04</servlet-name>
<servlet-class>myServlet.HelloServlet04</servlet-class>
<load-on-startup>2</load-on-startup>
</servlet>
8.ServletConfig:
Servlet 的配置,通过这个对象,可以获取 servlet 在配置的时候一些信息。
public class HelloServletConfig extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// TODO 自动生成的方法存根
//1.得到servlet配置对象,专门用于在配置servlet的信息
ServletConfig config = getServletConfig();
/*
* 方法一:String getServletName():获取配置文件中的名称
*/
String ServletName = config.getServletName();
System.out.println(ServletName);
/*
* 方法二:String getInitParameter(name):获取配置文件里面初始名为name的值
*/
String InitParameter = config.getInitParameter("address");
System.out.println(InitParameter);
/*
* 方法三:Enumeration<String> getInitParameterNames():获取配置文件里面所有初始变量名
*/
Enumeration<String> InitParameterNames = config.getInitParameterNames();
while (InitParameterNames.hasMoreElements()) {
String name = (String) InitParameterNames.nextElement();
String value = config.getInitParameter(name);
System.out.println("name=="+name+" value=="+value);
}
}
//就算是post请求也执行get的方法
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// TODO 自动生成的方法存根
doGet(req, resp);
}
}
配置文件如下:
<servlet>
<servlet-name>HelloServletConfig</servlet-name>
<servlet-class>myServlet.HelloServletConfig</servlet-class>
<!-- 可以添加初始化参数 -->
<init-param>
<param-name>name</param-name>
<param-value>zhujinsong</param-value>
</init-param>
<init-param>
<param-name>address</param-name>
<param-value>Beijing</param-value>
</init-param>
<init-param>
<param-name>age</param-name>
<param-value>18</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>HelloServletConfig</servlet-name>
<url-pattern>/HelloServletConfig</url-pattern>
</servlet-mapping>
9.使用Servlet的原因:
-
未来我们自己开发的一些应用,使用到了一些技术,或者一些代码,我们不会。但是有人写出来了。它的代码放置在了自己的 servlet 类里面。
-
刚好这个 servlet 里面需要一个数字或者叫做变量值。但是这个值不能是固定了。所以要求使用到这个 servlet 的公司,在注册 servlet 的时候,必须要在 web.xml 里面,声明 init-params。