【学习笔记】HttpServletResponse
web服务器接收到客户端的http请求,针对这个请求,分别创建一个代表请求的HttpServletRequest对象,和一个代表响应的HttpServletResponse对象
-
如果要获取客户端请求过来的参数:找HttpServletRequest
-
如果要给客户端一些响应信息:找HttpServletResponse
HttpServletResponse源码
HttpServletResponse 继承了ServletResponse
常量:状态码
int SC_CONTINUE = 100;
int SC_SWITCHING_PROTOCOLS = 101;
int SC_OK = 200;
int SC_CREATED = 201;
int SC_ACCEPTED = 202;
int SC_NON_AUTHORITATIVE_INFORMATION = 203;
int SC_NO_CONTENT = 204;
int SC_RESET_CONTENT = 205;
int SC_PARTIAL_CONTENT = 206;
int SC_MULTIPLE_CHOICES = 300;
int SC_MOVED_PERMANENTLY = 301;
int SC_MOVED_TEMPORARILY = 302;
int SC_FOUND = 302;
int SC_SEE_OTHER = 303;
int SC_NOT_MODIFIED = 304;
int SC_USE_PROXY = 305;
int SC_TEMPORARY_REDIRECT = 307;
int SC_BAD_REQUEST = 400;
int SC_UNAUTHORIZED = 401;
int SC_PAYMENT_REQUIRED = 402;
int SC_FORBIDDEN = 403;
int SC_NOT_FOUND = 404;
int SC_METHOD_NOT_ALLOWED = 405;
int SC_NOT_ACCEPTABLE = 406;
int SC_PROXY_AUTHENTICATION_REQUIRED = 407;
int SC_REQUEST_TIMEOUT = 408;
int SC_CONFLICT = 409;
int SC_GONE = 410;
int SC_LENGTH_REQUIRED = 411;
int SC_PRECONDITION_FAILED = 412;
int SC_REQUEST_ENTITY_TOO_LARGE = 413;
int SC_REQUEST_URI_TOO_LONG = 414;
int SC_UNSUPPORTED_MEDIA_TYPE = 415;
int SC_REQUESTED_RANGE_NOT_SATISFIABLE = 416;
int SC_EXPECTATION_FAILED = 417;
int SC_INTERNAL_SERVER_ERROR = 500;
int SC_NOT_IMPLEMENTED = 501;
int SC_BAD_GATEWAY = 502;
int SC_SERVICE_UNAVAILABLE = 503;
int SC_GATEWAY_TIMEOUT = 504;
int SC_HTTP_VERSION_NOT_SUPPORTED = 505;
负责向浏览器发送数据的方法:
ServletOutputStream getOutputStream() throws IOException; //其他
PrintWriter getWriter() throws IOException; //发送中文
负责向浏览器发送响应头的方法:
void setCharacterEncoding(String var1);
void setContentLength(int var1);
void setContentLengthLong(long var1);
void setContentType(String var1);
void setDateHeader(String var1, long var2);
void addDateHeader(String var1, long var2);
void setHeader(String var1, String var2);
void addHeader(String var1, String var2);
void setIntHeader(String var1, int var2);
void addIntHeader(String var1, int var2);
void setStatus(int var1);
...
HttpServletResponse应用
像浏览器输出消息
resp.getWriter().print("hello,world");
下载文件
在网页中请求一个地址,就会下载一个文件
下载步骤:
-
要获取下载文件的路径
-
下载的文件名
-
设置支持下载文件的头消息
-
获取下载文件的输入流
-
创建缓冲区
-
获取OutputStream对象
-
将FileOutputStream流写入到缓冲区,使用OutputStream将缓冲区中的对象输出到客户端
package com.wang;
import jakarta.servlet.ServletException;
import jakarta.servlet.ServletOutputStream;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.FileInputStream;
import java.io.IOException;
public class Response1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//1.获取下载文件的路径
String filePath = "E:\\Growth\\code\\JavaWeb\\javaweb-02-servlet\\response\\target\\classes\\1.png";
//2.下载的文件名
String fileName = filePath.substring(filePath.lastIndexOf("\\") + 1);
//3.设置支持下载文件的头消息
resp.setHeader("Content-Disposition","attachment;filename=" + fileName);
//4.获取下载文件的输入流
FileInputStream fis = new FileInputStream(filePath);
//5.创建缓冲区
int len = 0;
byte[] buffer = new byte[1024];
//6.获取OutputStream对象
ServletOutputStream sos = resp.getOutputStream();
//7.将流写入到缓冲区,使用OutputStream将缓冲区中的对象输出到客户端
while ((len=fis.read(buffer))!=-1){
sos.write(buffer,0,len);
}
//8.关闭流
sos.close();
fis.close();
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
问题:
-
关于文件路径问题:
代码中已经将文件路径写死,但是ServletContext提供了一个方法getRealPath()用来获取文件的路径,但是结果为
E:\JAVA\apache-tomcat-10.0.26\webapps\response_war\1.png
也就是web的资源文件夹下,但是我们的文件没有放在这个文件夹,
而是在"E:\Growth\code\JavaWeb\javaweb-02-servlet\response\target\classes\1.png"
解决的办法:
URL resource = this.getClass().getClassLoader().getResource("1.png"); String filePath = resource.toString();
调用这个类的类加载器中的方法 getResource(),得到的是
file:/E:/JAVA/apache-tomcat-10.0.26/webapps/response_war/WEB-INF/classes/1.png
tomcat一运行就把文件加载到该文件夹,所以我们使用这种方式是没有错误的,至于将文件路径写死,不可取。
并且写死的文件夹也不是我们放资源的文件夹,而是target文件夹,是运行之后自动生成的文件夹
-
关于把文件夹修改为文件名
首先有一个问题是为什么要把文件名单独分类出来?
-
为了保持它原先的文件名(这个原因无关紧要)
-
必须要和它原先的文件的后缀相同,也就是文件名必须以.png为结尾,否则下载下来就不是图片文件了。
经过几次实验,发现我们手动把文件名改为xxx.png,也是可以的。
第二个问题把文件名从文件路径中分离出来的方式?
用到了两个方法:
-
subString -- 截取字符串
-
new String("this is text").subString(3,10) --结果:s is te [3,10)
-
new String("this is text").subString(3) --结果 s is text
-
-
lastIndexOf --返回字符串最后出现的位置
-
new String("this is text").lastIndexOf("e") --结果 10
-
我们看文件名
file:/E:/JAVA/apache-tomcat-10.0.26/webapps/response_war/WEB-INF/classes/1.png
lastindexOf("/") -- 结果是 72
s.substring(s.lastIndexOf("/") + 1);
就是从"/" 往后所有的字符串都截下来,也就是文件名了。
-
-
关于setHeader()
设置Http消息头,参数都是字符串类型
下载文件:response.setHeader( "Content-Disposition", "attachment;filename=" +name)
其中Content-Disposition 表示要将服务器发送过来的数据保存到磁盘
attachment;filename= 是 设置文件的名称
在上面的例子中,文件的名字不能是中文,因为会乱码,所以我们需要设置一下字符编码
resp.setHeader("Content-Disposition","attachment;filename=" + URLEncoder.encode(fileName,"UTF-8"));
使用URLEncoder对象的encode方法,将文件名的编码设置为UTF-8
实现重定向
重定向:一个web资源收到客户端请求后,它会通知客户端去访问另一个web资源,这个过程叫做重定向。
常见的重定向场景:需要登录时才能使用其他功能时,会重定向到登录页面。
方法:sendRedirect()
参数是,要跳转的地址,注意这个地址包括项目的地址,不能只是要跳转的资源的地址
package com.wang;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
public class RedirectResponse extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.sendRedirect("/r/test");//需要加上项目的地址
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
结果:
访问的是:localhost:8080/r/red
最后跳转到了 localhost:8080/r/test
我们可以发现请求的状态码是302
响应头中的location为 /r/test
sendRedirect的原理就是如下,设置Location 和状态码
resp.setHeader("Location");
resp.setStatus(302);
笔试题:重定向和请求转发的区别?
相同点:
-
页面都可以跳转
不同点:
-
请求转发,url不会发生变化
-
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!