写在开头:本文是学习Java后端开发的个人笔记,便于自己复习。文章内容引用了尚硅谷的javaweb教学,有兴趣的朋友可以上B站搜索。
Servlet
Ⅰ Servlet技术
Ⅱ ServletConfig类
Ⅲ ServletContext类
Ⅳ HTTP协议
Ⅴ HttpServletRequest类
Ⅵ HttpServletResponse类
Ⅰ Servlet技术
-
什么是servlet
- Servlet 是 JavaEE 规范之一。规范就是接口
- Servlet 就 JavaWeb 三大组件之一。三大组件分别是:Servlet 程序、Filter 过滤器、Listener 监听器。
- Servlet 是运行在服务器上的一个 java 小程序,它可以接收客户端发送过来的请求,并响应数据给客户端。 -
手动实现第一个Servlet程序
实现一个Servlet程序,一般要经过如下几个步骤:
- 编写一个类去实现 Servlet 接口
- 重写service 方法,处理请求,并响应数据
- 到 web.xml 中去配置 servlet 程序的访问地址
接下来我们开始实现我们的第一个Servlet程序:首先我们编写一个类HelloServlet派生于Servlet,并重写其service方法。为了让我们更好理解servlet的生命周期,我们在相应的方法上打印标记。public class HelloServlet implements Servlet { public HelloServlet() { System.out.println("1、构造方法"); } @Override public void init(ServletConfig servletConfig) throws ServletException { System.out.println("2、init初始化方法"); } @Override public ServletConfig getServletConfig() { return null; } @Override public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException { System.out.println("3、service方法 === Hello Servlet 被访问了"); // 类型转换 (因为它有getMethod()方法) HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest; // 获取请求的方式 String method = httpServletRequest.getMethod(); System.out.println("本次请求方式为: " + method); } @Override public String getServletInfo() { return null; } @Override public void destroy() { System.out.println("4、destroy销毁方法"); } }
接下来到 web.xml 中去配置 servlet 程序的访问地址
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd" version="4.0"> <!-- servlet 标签给 Tomcat 配置 Servlet 程序 --> <servlet> <!--servlet-name 标签 Servlet 程序起一个别名(一般是类名) --> <servlet-name>HelloServlet</servlet-name> <!--servlet-class 是 Servlet 程序的全类名--> <servlet-class>com.jarreet.servlet.HelloServlet</servlet-class> </servlet> <!--servlet-mapping 标签给 servlet 程序配置访问地址--> <servlet-mapping> <!--servlet-name 标签的作用是告诉服务器,我当前配置的地址给哪个 Servlet 程序使用--> <servlet-name>HelloServlet</servlet-name> <!--url-pattern 标签配置访问地址 <br/> / 斜杠在服务器解析的时候,表示地址为:http://ip:port/工程路径 <br/> /hello 表示地址为:http://ip:port/工程路径/hello <br/> --> <url-pattern>/hello</url-pattern> </servlet-mapping> </web-app>
写一个表单来提交资源请求
<html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <form action="http://localhost:8080/06_servlet_war_exploded/hello" method="get"> <input type="submit"> </form> </body> <html>
我们来运行这个工程,进入a.html。并点击提交按钮
提交完后,页面获取资源,servlet程序开始执行
重复点击提交按钮
停止运行
由上边的结果我们可以看出来Servlet的生命周期大致分为分为四个阶段:
- 执行 Servlet 构造器方法
- 执行 init 初始化方法
- 执行 service 方法
- 执行 destroy 销毁方法
值得注意的是:第一、二步,是在第一次访问的时候创建 Servlet 程序会调用;第三步,每次访问都会调用;第四步,在 web 工程停止的时候调用。url地址到Servlet程序的访问原理如下图:
-
通过继承 HttpServlet 实现 Servlet 程序
一般在实际项目开发中,都是使用继承 HttpServlet 类的方式去实现 Servlet 程序。其步骤如下:
- 编写一个类去继承 HttpServlet 类
- 根据业务需要重写 doGet 或 doPost 方法
- 到 web.xml 中的配置 Servlet 程序的访问地址Servlet 类的代码:
public class HelloServlet2 extends HttpServlet { // doGet()在 get 请求的时候调用 @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("HelloServlet2 的 doGet 方法"); } // doPost()在 post 请求的时候调用 @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("HelloServlet2 的 doPost 方法"); } }
web.xml 中的配置:
<servlet> <servlet-name>HelloServlet2</servlet-name> <servlet-class>com.atguigu.servlet.HelloServlet2</servlet-class> </servlet> <servlet-mapping> <servlet-name>HelloServlet2</servlet-name> <url-pattern>/hello2</url-pattern> </servlet-mapping>
-
使用 IDEA 创建 Servlet 程序
菜单:new ->Servlet 程序
配置 Servlet 的信息:
-
整个Servlet类的继承体系
Ⅱ ServletConfig类
ServletConfig 类的三大作用
- 可以获取 Servlet 程序的别名 servlet-name 的值
- 获取初始化参数 init-param
- 获取 ServletContext 对象
注意:重写init方法时,一定要调用父类的init(ServletConfig)操作
web.xml 中的配置:
<!-- servlet 标签给 Tomcat 配置 Servlet 程序 -->
<servlet>
<!--servlet-name 标签 Servlet 程序起一个别名(一般是类名) -->
<servlet-name>HelloServlet</servlet-name>
<!--servlet-class 是 Servlet 程序的全类名-->
<servlet-class>com.atguigu.servlet.HelloServlet</servlet-class>
<!--init-param 是初始化参数-->
<init-param>
<!--是参数名-->
<param-name>username</param-name>
<!--是参数值-->
<param-value>root</param-value>
</init-param>
<init-param>
<!--是参数名-->
<param-name>url</param-name>
<!--是参数值-->
<param-value>jdbc:mysql://localhost:3306/project1</param-value>
</init-param>
</servlet>
<!--servlet-mapping 标签给 servlet 程序配置访问地址-->
<servlet-mapping>
<!--servlet-name 标签的作用是告诉服务器,我当前配置的地址给哪个 Servlet 程序使用-->
<servlet-name>HelloServlet</servlet-name>
<!--
url-pattern 标签配置访问地址
/ 斜杠在服务器解析的时候,表示地址为:http://ip:port/工程路径
/hello 表示地址为:http://ip:port/工程路径/hello
-->
<url-pattern>/hello</url-pattern>
</servlet-mapping>
servlet中的代码:
@Override
public void init(ServletConfig servletConfig) throws ServletException{
System.out.println("2、init初始化方法");
//1、可以获取 Servlet 程序的别名 servlet-name 的值
System.out.println("HelloServlet程序的别名是:" + servletConfig.getServletName());
//2、获取初始化参数 init-param
System.out.println("初始化参数username的值是:" + servletConfig.getInitParameter("username"));
System.out.println("初始化参数url的值是:" + servletConfig.getInitParameter("url"));
//3、获取 ServletContext 对象
System.out.println(servletConfig.getServletContext());
}
运行结果:
Ⅲ ServletContext类
-
什么是ServletContext类
- ServletContext 是一个接口,它表示 Servlet 上下文对象
- 一个 web 工程,只有一个 ServletContext 对象实例。
- ServletContext 对象是一个域对象。
- ServletContext 是在 web 工程部署启动的时候创建。在 web 工程停止的时候销毁。域对象,是可以像 Map 一样存取数据的对象,叫域对象。这里的域指的是存取数据的操作范围,整个 web 工程。
-
ServletContext 类的四个作用
- 获取 web.xml 中配置的上下文参数 context-param
- 获取当前的工程路径,格式: /工程路径
- 获取工程部署后在服务器硬盘上的绝对路径
- 像 Map 一样存取数据
web.xml 中的配置:
<!--context-param 是上下文参数(它属于整个 web 工程)-->
<context-param>
<param-name>username</param-name>
<param-value>context</param-value>
</context-param>
<context-param>
<param-name>password</param-name>
<param-value>abc123</param-value>
</context-param>
ServletContext 演示代码1:
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
ServletContext context = getServletConfig().getServletContext();
//1、获取 web.xml 中配置的上下文参数 context-param
String username = context.getInitParameter("username");
String password = context.getInitParameter("password");
System.out.println("context-param参数username的值是:" + username);
System.out.println("context-param参数password的值是:" + password);
//2、获取当前的工程路径,格式: /工程路径
System.out.println("当前工程路径:" + context.getContextPath());
//3、获取工程部署后在服务器硬盘上的绝对路径
System.out.println("当前工程部署的绝对路径是:" + context.getRealPath("/"));
/* / 斜杠被服务器解析地址为:http://ip:port/工程名/ 映射到 IDEA 代码的 web 目录;因此我们可以通过getRealPath取得web工程下文件的路径 */
}
ServletContext 演示代码2:
ContextServlet1代码(配置文件自行编写)
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 获取 ServletContext 对象
ServletContext context = getServletContext();
System.out.println(context);
System.out.println("保存之前:Context1 获取key1的值是:" + context.getAttribute("key1"));
context.setAttribute("key1","value");
System.out.println("Context1中获取域数据key1的值是:" + context.getAttribute("key1"));
}
ContextServlet2代码(配置文件自行编写)
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
ServletContext context = getServletContext();
System.out.println(context);
System.out.println("Context2 中获取的数据key1的值是:" + context.getAttribute("key1"));
}
运行结果:
从上边这个例子,我们可以很明显的看出一个web工程只有一个ServletContext实例,因为contex1和contex2的地址都是相同的。并且只要将数据存储以后,整个web工程都可以取出数据。
Ⅳ HTTP协议
-
什么是HTTP协议
所谓 HTTP 协议,就是指,客户端和服务器之间通信时,发送的数据,需要遵守的规则,叫 HTTP 协议。 HTTP 协议中的数据又叫报文。 -
请求的HTTP协议格式
客户端给服务器发送数据叫请求。 服务器给客户端回传数据叫响应。 请求又分为 GET 请求,和 POST 请求两种
(1) GET 请求
- 请求行
+ 请求的方式
+ 请求的资源路径[+?+请求参数]
+ 请求的协议的版本号
- 请求头
+ key : value 组成 不同的键值对,表示不同的含义。
(2)POST请求
- 请求行
+ 请求的方式
+ 请求的资源路径[+?+请求参数]
+ 请求的协议的版本号
- 请求头
+ key : value 组成 不同的键值对,表示不同的含义。
- 请求体
+ 发给服务器的数据
(3)常用请求头的说明
Accept: 表示客户端可以接收的数据类型
Accpet-Languege: 表示客户端可以接收的语言类型
User-Agent: 表示客户端浏览器的信息
Host: 表示请求时的服务器 ip 和端口号(4)哪些是 GET 请求,哪些是 POST 请求
- GET 请求有哪些:
+ form 标签 method=get
+ a 标签
+ link 标签引入 css
+ Script 标签引入 js 文件
+ img 标签引入图片
+ iframe 引入 html 页面
+ 在浏览器地址栏中输入地址后敲回车
- POST 请求有哪些:
+ form 标签 method=post -
响应的HTTP协议格式
- 响应行
+ 响应的协议和版本号
+ 响应的状态码
+ 响应状态描述符
- 响应头
+ key : value 不同的响应头,有其不同含义
- 响应体
+ 是回传给客户端的数据
-
常用的响应码与MIME类型说明
(1)常用的响应码
200 表示请求成功
302 表示请求重定向
404 表示请求服务器已经收到了,但是你要的数据不存在(请求地址错误)
500 表示服务器已经收到请求,但是服务器内部错误(代码错误)(2)MIME类型说明
MIME 是 HTTP 协议中数据类型。
MIME 的英文全称是"Multipurpose Internet Mail Extensions" 多功能 Internet 邮件扩充服务。MIME 类型的格式是“大类型/小 类型”,并与某一种文件的扩展名相对应。常见的 MIME 类型:
文件 MIME 类型 超文本标记语言文本 .html , .htm text/html 通文本 .txt text/plain RTF 文本 .rtf application/rtf GIF 图形 .gif image/gif JPEG 图形 .jpeg,.jpg image/jpeg au 声音文件 .au audio/basic MIDI 音乐文件 mid,.midi audio/midi,audio/x-midi RealAudio 音乐文件 .ra, .ram audio/x-pn-realaudio MPEG 文件 .mpg,.mpeg video/mpeg AVI 文件 .avi video/x-msvideo GZIP 文件 .gz application/x-gzip TAR 文件 .tar application/x-tar
Ⅴ HttpServletRequest类
-
HttpServletRequest类的作用
每次只要有请求进入 Tomcat 服务器,Tomcat 服务器就会把请求过来的 HTTP 协议信息解析好封装到 Request 对象中。 然后传递到 service 方法(doGet 和 doPost)中给我们使用。我们可以通过 HttpServletRequest 对象,获取到所有请求的 信息。 -
HttpServletRequest 类的常用方法
方法 作用 getRequestURI() 获取请求的资源路径 getRequestURL() 获取请求的统一资源定位符(绝对路径) getRemoteHost() 获取客户端的 ip 地址 getHeader() 获取请求头 getParameter() 获取请求的参数 getParameterValues() 获取请求的参数(多个值的时候使用) getMethod() 获取请求的方式 GET 或 POST setAttribute(key, value) 设置域数据 getAttribute(key) 获取域数据 getRequestDispatcher() 获取请求转发对象 常用API示例代码:
public class RequestAPIservlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 获取请求的资源路径 System.out.println("URI = " + req.getRequestURI()); // 获取请求的统一资源定位符(绝对路径) System.out.println("URL = " + req.getRequestURL()); // 获取客户端的 ip 地址 /** * 在 IDEA 中,使用 localhost 访问时,得到的客户端 ip 地址是 ===>>> 127.0.0.1 * 在 IDEA 中,使用 127.0.0.1 访问时,得到的客户端 ip 地址是 ===>>> 127.0.0.1 * 在 IDEA 中,使用 真实 ip 访问时,得到的客户端 ip 地址是 ===>>> 真实的客户端 ip 地址 */ System.out.println("客户端ip地址 = " + req.getRemoteHost()); } // 获取请求头 System.out.println("请求头User-Agent = " + req.getHeader("User-Agent")); // 获取请求的方式 GET 或 POST System.out.println("请求的方式 = " + req.getMethod()); }
运行结果:
-
如何获取请求参数
表单<form action="http://localhost:8080/07_servlet/param" method="get"> 用户名:<input type="text" name="username"><br/> 密码:<input type="password" name="password"><br/> <input type="checkbox" name="hobby" value="cpp">c++ <input type="checkbox" name="hobby" value="java">Javainput type="checkbox" name="hobby" value="js">JavaScript<br/> <input type="submit"> </form>
java代码
public class ParameterServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 获取请求参数 String username = req.getParameter("username"); String password = req.getParameter("password"); String[] hobbys = req.getParameterValues("hobby"); System.out.println("用户名:" + username); System.out.println("密码:" + password); System.out.println("兴趣爱好:" + Arrays.asList(hobbys)); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 设置请求体的字符集为 UTF-8,从而解决 post 请求的中文乱码问题 req.setCharacterEncoding("UTF-8"); // 获取请求参数 String username = req.getParameter("username"); String password = req.getParameter("password"); String[] hobbys = req.getParameterValues("hobby"); System.out.println("用户名:" + username); System.out.println("密码:" + password); System.out.println("兴趣爱好:" + Arrays.asList(hobbys)); } }
运行结果:
-
请求的转发
请求转发是指,服务器收到请求后,从一次资源跳转到另一个资源的操作叫请求转发。
举个例子,现在有两个Servlet程序:servlet1和sevlet2;
我们请求servlet1并发送数据,servlet1会接收请求的参数并存储一个“标记”到域中以便servlet2检验;接下来servlet1会获取走向servlet2的路径并将请求转发给servlet2。
servlet2接受到请求后依旧会查看请求的参数并在域中检查有没有servlet1留下的标记,获取标记后则开始处理自身的业务。
整个转发流程如下图:
servlet1代码:public class Servlet1 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 获取请求的参数(办事的材料)查看 String username = req.getParameter("username"); System.out.println("在Servlet1中查看参数:" + username); // 给材料 盖一个章,并传递到 Servlet2(柜台 2)去查看 req.setAttribute("key1","柜台1的章"); // 问路:Servlet2(柜台 2)怎么走 /** * 请求转发必须要以斜杠打头,/ 斜杠表示地址为:http://ip:port/工程名/ , 映射到 IDEA 代码的 web 目录 */ RequestDispatcher requestDispatcher = req.getRequestDispatcher("/servlet2"); // 走向 Sevlet2(柜台 2) requestDispatcher.forward(req, resp); } }
servlet2代码:
public class Servlet2 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 获取请求的参数(办事的材料)查看 String username = req.getParameter("username"); System.out.println("在Servlet2中查看参数:" + username); // 查看 柜台 1 是否有盖章 Object key1 = req.getAttribute("key1"); System.out.println("柜台1是否有章:" + key1); // 处理自己的业务 System.out.println("Servlet2 处理自己的业务"); } }
运行结果:
向servlet1发送请求,参数usname=离渊
output
-
base标签的作用
base标签可以设置当前页面中所有相对路径工作时,参照哪个路径来进行跳转
举个例子:假设我们web工程下的a文件夹下的b文件夹下有个 c.html<body> 这是a下的b下的c.html页面 <a href="../../index.html">跳回首页</a> </body>
而web工程下的index.html如下:
<body> 这是web下的index.html<br/> <a href="http://localhost:8080/07_servlet/forwardc">请求转发:a/b/c.html</a> </body>
frowardc代码如下:
public class ForwardC extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("经过了Forward程序"); req.getRequestDispatcher("/a/b/c.html").forward(req,resp); } }
正常来说,我们点击index.html的 请求转发:a/b/c.html 会请求资源forwardc 并跳转到 c.html ,再 点击 c.html 的跳回首页 能回到 index。但事实上上述代码只能从index跳转到c,不能从c 跳转回index。
这是为什么呢?答案在下图:
若要顺利跳转回index,我们就要让路径正确,这时候就要用上base标签。<head> <meta charset="UTF-8"> <title>Title</title> <base href="http://localhost:8080/07_servlet/a/b/"> </head> <body> 这是a下的b下的c.html页面 <a href="../../index.html">跳回首页</a> </body>
这样路径就正确了
-
web中的路径和 / 斜杠 的意义
(1)Web 中的相对路径和绝对路径
相对路径是:
. 表示当前目录
… 表示上一层目录
… 表示当前目录/资源名绝对路径:
http://ip:port/工程路径/资源路径在实际开发中,路径都使用绝对路径,而不简单的使用相对路径。(绝对路径或base+相对路径)
(2)web 中 / 斜杠的不同意义
在 web 中 / 斜杠 是一种绝对路径。
/ 斜杠 如果被浏览器解析,得到的地址是:http://ip:port/
/ 斜杠 如果被服务器解析,得到的地址是:http://ip:port/工程路径
特殊情况: response.sendRediect(“/”); 把斜杠发送给浏览器解析。得到 http://ip:port/
Ⅵ HttpServletResponse类
-
HttpServletResponse类的作用
HttpServletResponse 类和 HttpServletRequest 类一样。每次请求进来,Tomcat 服务器都会创建一个 Response 对象传 递给 Servlet 程序去使用。HttpServletRequest 表示请求过来的信息,HttpServletResponse 表示所有响应的信息
我们如果需要设置返回给客户端的信息,都可以通过 HttpServletResponse 对象来进行设置 -
往客户端回传数据
(1)两个输出流
字节流 getOutputStream(); 常用于下载(传递二进制数据)
字符流 getWriter(); 常用于回传字符串(常用)两个流同时只能使用一个;使用了字节流,就不能再使用字符流,反之亦然,否则就会报错。
(2)回传数据的方式
public class ResponseIOservlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 往客户端回传 字符串 数据 PrintWriter writer = resp.getWriter(); writer.write("respone's content!"); } }
(3)响应的乱码解决
当服务器向浏览器回传中文字符串时,由于响应对象的编码和浏览器编码不是UTF-8,因此会出现乱码。这时我们只要通过setContentType()方法,将字符集设置为UTF-8就行了public class ResponseIOservlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 它会同时设置服务器和客户端都使用 UTF-8 字符集,还设置了响应头 // 此方法一定要在获取流对象之前调用才有效 resp.setContentType("text/html; charset=UTF-8"); // 往客户端回传 中文字符串 数据 PrintWriter writer = resp.getWriter(); writer.write("离渊不破,百川成海!"); } }
(4)请求重定向
请求重定向,是指客户端给服务器发请求,然后服务器告诉客户端说。我给你一些地址。你去新地址访问。叫请求 重定向(因为之前的地址可能已经被废弃)
请求重定向的两种方案:- 第一种方案:
// 设置响应状态码 302 ,表示重定向,(已搬迁) resp.setStatus(302); // 设置响应头,说明 新的地址在哪里 resp.setHeader("Location", "http://localhost:8080");
- 第二种方案(推荐):
resp.sendRedirect("http://localhost:8080");
以第二种方法举例:
Response1:public class Response1 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("到达 Response1"); resp.sendRedirect("http://localhost:8080/07_servlet/response2"); } }
Response2:
public class Response2 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("到达 response2"); resp.getWriter().write("response2's result"); } }
运行结果