HTTP协议:
- 请求消息:客户端发送给服务器端的数据
数据格式:
1. 请求行
2. 请求头
3. 请求空行
4. 请求体 - 响应消息:服务器端发送给客户端的数据
- 响应行
- 响应头
- 响应空行
- 响应体
例:先创建一个服务器来进行测试:
新建一个javaee项目,
更改index.jsp里面的内容:
启动成功后,会自动打开网页:
打开F12,刷新页面,然后,点击第一条信息,
响应头如下:
响应体如下:
注:响应行是在响应头的第一行
例:上面的例子中
响应行:HTTP/1.1 200:
组成:协议/版本 响应状态码 状态码描述
响应状态码:是服务器告诉客户端你给我的请求和响应的状态怎么样了。
分类:- 1xx:客户端给服务器发消息,发了一半,服务器端就等着,等了一会就给客户端发了1xx状态码,询问客户端消息发完没有。
- 2xx:这次请求和响应是成功的。代表:200
- 3xx:重定向。代表:302(重定向)304(访问缓存)
重定向分析:浏览器请求a资源,a发现自己办不了,返回浏览器一个302(重定向)并给了一个c的地址,让浏览器去找c,这时候,浏览器就去找c了。
访问缓存分析:首先浏览器第一次访问服务器,假设访问的是图片,服务器给浏览器返回图片,此时,浏览器会在本地把图片进行保存,但浏览器第二次进行访问服务器上该图片的时候,服务器会进行判断,如果发现图片没有发生改变,就会给浏览器返回一个304告诉浏览器去自己本地上找。
右边Not Modified:就是代表没有变化。 - 4xx:代表失败,代表客户端错误。代表客户端请求的资源服务器端没有。例子:404代表请求的路径资源没有,405代表请求方式没有对应doxxx方法。
405错误举例:先创建一个servlet的实现类,把doGet方法给删掉,
再进行访问对应的路径:
这样就创建出来405的错误了。 - 5xx:代表失败,代表服务器端错误。代表500(服务器内部出现异常,服务器内部报错了,会把错误信息展现再页面上)
响应头: - 格式:头名称: 值
- 常见的响应头:
Content-Type:text/html;charset=UTF-8
1. Content-Type:服务器告诉客户端本次响应体数据格式以及编码格式
2. Content-disposition:服务器告诉客户端以什么格式打开响应体数据
值:
in-line:默认值,在当前页面内打开
attachment;filename=xxx:以附件形式打开响应体。文件下载
响应体:(就是html页面)
- 设置响应体:
使用步骤:
1.获取输出流
字符输出流:PrintWriter getWriter()
字节输出流:ServletOutputStream getOutputStream()
2.使用输出流,将数据输出到客户端浏览器
案例:
1.完成重定向(资源跳转的方式)
2.服务器输出字符数据到浏览器
3.服务器输出字节数据到浏览器
4.验证码
重定向:
6.测试实验:
编写代码:如果访问demo的时候,可以跳转到demo2上面,结果如下:
注意:由于setStatus(302)中302基本不变,而且“localhost"也基本不变,所以,对于这两个方法可以改为:
下面这行代码也可以实现对应的功能。
重定向的特点:redirect
1.地址栏发生变化
2.重定向可以访问其他站点(服务器)的资源,比如百度
3. 重定向是两次请求。不能使用request对象来共享数据(两个request对象了)
转发的特点:forward
1. 转发地址栏路径不变
2. 转发只能访问当前服务器下的资源
3. 转发是一次请求,可以使用request对象来共享数据
面试常问:forward 和 redirect 的区别。
路径写法
1.路径分类:
1.相对路径:通过相对路径不可以确定唯一的资源。
如:./index.html代表的是当前路径下的index.html资源
你要先确定当前路径。
相对路径不以/开头,以.开头
注意:./是可以省略的 ,所以index.html代表的是当前路径
2.绝对路径:通过绝对路径可以确定唯一的资源
如:http://localhost/day15/demo
注意可以去点前面的ip和端口号写成:
/day15/demo
绝对路径可以简单认为,以/开头的
绝对路径使用规则:判断定义的路径给谁用的?
给客户端浏览器用:需要加虚拟目录(项目的访问路径)
给服务器使用:不需要加虚拟目录
绝对路径书写举例说明:
在web下面新建一个html页面,
启动并对其进行访问:
从这里可以看到,这很显然是给浏览器用的,而且他要给服务器进行发送信息,很显然要加虚拟路径。
但在服务器内部进行转发的时候,就不需要写虚拟目录了。
转发的实验:
写一个HttpServlet的实现类,实现doPost()当访问demo的时候进行转发。转发到/demo2(这里用的是绝对路径)
然后再实现/demo2下的HttpServlet实现类。但访问到/demo2的时候,控制台会输出post
测试如下:
可以看到,这里输出是post。而且,只有一次访问。
从机理来看,其实,当重定向的时候,绝对路径也是需要加虚拟目录的,因为他是两次访问,第二次也是从浏览器访问服务器的。
综上,只有在转发的时候绝对路径不用加虚拟目录。其他的时候都需要加的。
虚拟目录会进行变,所以不要写死
所以要将虚拟目录写成动态的。用方法来获取虚拟目录
方法如下:req.getContextPath();这是用来获取虚拟目录的
resp.sendRedirect(s + “/demo2”);在进行重定向操作。
这样就好了。
服务器输出字符数据到浏览器
步骤:
1.获取字符输出流
2.输出数据
代码如下:
部署方式改变,不要重新启动服务器
1.重新进行配置
里面这两个属性改成更改类和资源的时候变化
最后保存更改
最后进行debug启动
最后访问得如下结果:发现可以将数据进行写入。
用以上的方法会导致乱码
原因编码方式和解码方式不一样。
判断他们的编解码方式:先用ie访问这个网址,右键查看他的编码方式,发现默认是使用GBK(gb2312)(解码是GBK)
然后,再来看编码:
这个输入流对象是,获取到的,不是系统new出来的所以,如果是系统自己new由于我自己电脑是中文操作系统,则也是GBK编码,但是他是获取到的,他是有tomcat创造出来的,所以他的编码方式是ISO-8859-1
所以,我们设置流的编码为gbk,在获取流之前,设置流的默认编码为GBK这样结果就好了。
但是,这样也有问题,就是你得实现知道浏览器是什么解码方式:
要是,但设置服务器编码格式的时候,再告诉浏览器以什么方式进行解码,这样是最好的。resp中也提供了对应的方法。
再次用ie进行访问,发现编码方式是utf-8了,这就确保无论怎么样也不会乱码。
上面代码第一行可以省略,第二行已经代表了第一行了,因为第二行中的content-type是固定的,所以tomcat提供了一个简单的方法:
服务器输出字节数据到浏览器
步骤和输出字符差不多
1.获取字节输出流
2.输出数据
(一般用来输出图片的)
代码如下:
测试如下:
ServletContext对象
概念:代表整个web应用,可以和程序的容器(服务器)来通信。
获取:
1.通过request对象来获取
request.getServletContext();
2.通过HttpServlet获取
this.getServletContext();
功能:
1.获取MIME类型:
MIME类型:在互联网通信过程中定义的一种文件数据类型
格式: 大类型/小类型 例:text/html
方法 :String getMimeType(String file)
实验结果如下:
2.域对象:共享数据的(req也是域对象)
所有域对象都有以下的方法
1. setAttribute(String name,Object value)
2. getAttribute(String name)
3. removeAttribute(String name)
测试如下:先写一个HttpServlet实现类,然后在该类上对ServletContext进行写入操作。
然后再写一个路径是/demo4的HttpServlet的实现类,尝试取出数据看看能不能取出来。
经过测试:先访问/demo3再访问/demo4发现可以取出数据,故测试成功。
注意:该对象一般是不怎么使用的,第一点:因为大家都可以访问,不安全。第二点:因为这个对象服务器一创建就创建,服务器关闭才关闭,在内存中停留时间比较久,所以,比较占内存,故一般来说也不使用。
3.获取文件的真实(服务器路径)路径
1.方法:String getRealPath(String path)
实验分析:要找到下图上的三个txt文件
首先先访问b.txt文件,注意,web下面的路径是"/b.txt"可以来决定,决定路径就是在web下面的,故可以这么访问:
输出结果如下:F:\niagara demo\day5-17\out\artifacts\Response_study_war_exploded\b.txt
根据输出结果:去本地进行查找得到如下结论:
现在访问c.txt 由于项目是从web路径下开始的。所以,访问c.txt就可以用"WEB-INF/c.txt"来进行访问。结果如下:
现在访问a.txt,注意a.txt文件时放在src目录下,src目录下的所有文件到时候都会放在WEB-INF目录下:的class文件下
所以,src目录就等效成/WEB-INF/class/
所以,访问a.txt目录就有写成:
访问结果如下:
有了这个路径就可以加载文件到内存了。
综合案例:文件下载:
1.页面显示超链接
2.点击超链接后弹出下载提示框
3.完成图片的下载
流程图如下:
故先创建一个网页:
在创建一个HttpServlet的实现类代码如下:
package cn.sainan114;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.FileInputStream;
import java.io.IOException;
/**
* FileName: Case
*
* @Author:luguobao Date: 16:20
* Description:
* History:
* <author> <time> <version> <desc>
* 作者姓名 修改时间 版本号 描述
*/
@WebServlet("/case")
public class Case extends HttpServlet {
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//1.先获取参数
String filename = req.getParameter("filename");
//2.根据参数来获取在服务器上的绝对路径
ServletContext servletContext = this.getServletContext();
//因为1.jpg 和2.jpg都直接在web下
String realPath = servletContext.getRealPath("/" + filename);
//设为字符输入流
FileInputStream fis = new FileInputStream(realPath);
//设置输出的响应头
//设置响应头的类型:content-type
String mimeType = servletContext.getMimeType(filename);
resp.setHeader("content-type",mimeType);
//设置相应头的打开方式:content-disposition
resp.setHeader("content-disposition","attachment;filename" + filename);
//将输入流的数据写到输出流中
ServletOutputStream sos = resp.getOutputStream();
//新建一个水桶来存水
byte[] buff = new byte[1024*8];
int len = 0;
//但fis数据不为空的时候,写数据一次读取1024*8个数据
while ((len = fis.read(buff))!=-1){
//写1024*8个数据
sos.write(buff,0,len);
}
//关闭输入流
fis.close();
}
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
this.doPost(req,resp);
}
}
进行访问:结果报错,报错结果404;显然路径错了,经过检查发现虚拟目录没加。
故将test.html改为如下:
下面重新访问:发现报错500(服务器出现问题)
F:\niagara demo\day5-17\out\artifacts\Response_study_war_exploded\img\1.jpg (系统找不到指定的文件。)报错内容说:服务器上找找不到文件,
我去找了以下,发现确实没有该图片理论上复制在web下面就应该有这个图片的,具体原因还没发现。但是由于有2.jpg所以图片2是可以下载的。