- Servlet是JavaEE规范之一,规范就是接口
- Servlet是JavaWeb三大组件之一,三大组件是:Servlet程序、Filter过滤器、Listener监听器
- Servlet是运行在服务器上的一个java小程序,它可以接受客户端发送过来的请求,并响应数据给客户端
:::tips
注意:如果第一次使用的时候,发现没有Servlet这个接口,说明我们没有导入servlet这个jar包
package com.zanedu.servlet;
import jakarta.servlet.*;
import java.io.IOException;
public class HelloServlet implements Servlet {
/**
* service 方法是专门用来处理请求和响应的
* @param servletRequest
* @param servletResponse
* @throws ServletException
* @throws IOException
*/
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
System.out.println("Hello Servlet 被访问了");
}
}
<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>
<servlet-name>HelloServletservlet-name>
<servlet-class>com.zanedu.servlet.HelloServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>HelloServletservlet-name>
<url-pattern>/hellourl-pattern>
servlet-mapping>
web-app>
:::tips
第一、二步,是在第一次访问的时候,创建Servlet程序会调用,只会调用一次
第三步,每次访问都会调用,即会被调用多次
package com.zanedu.servlet;
import jakarta.servlet.*;
import jakarta.servlet.http.HttpServletRequest;
import java.io.IOException;
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;
}
/**
* service 方法是专门用来处理请求和响应的
* @param servletRequest
* @param servletResponse
* @throws ServletException
* @throws IOException
*/
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
System.out.println("3 service === Hello Servlet 被访问了");
}
@Override
public String getServletInfo() {
return null;
}
@Override
public void destroy() {
System.out.println("4 destroy销毁方法");
}
}
//类型转换,将ServletRequest转换成HttpServletRequest
//因为它有getMethod()方法,而ServletRequest父接口中没有这个方法
HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;
//获取请求的方式
String method = httpServletRequest.getMethod();
//类型转换(因为它有getMethod()方法,而servletRequest是父接口,没有这个方法,因此要转换一下)
HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;
//获取请求的方式
String method = httpServletRequest.getMethod();
if ("GET".equals(method)) {
System.out.println("get请求");
} else if ("POST".equals(method)) {
System.out.println("post请求");
}
/**
* 做get请求的操作
*/
public void doGet() {
System.out.println("get请求");
System.out.println("get请求");
}
/**
* 做post请求的操作
*/
public void doPost() {
System.out.println("post请求");
System.out.println("post请求");
}
package com.zanedu.servlet;
import jakarta.servlet.ServletConfig;
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 HelloServlet2 extends HttpServlet {
/**
* doGet() 方法在get请求的时候调用
* @param req
* @param resp
* @throws ServletException
* @throws IOException
*/
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("HelloServlet2 的doGet方法被调用");
}
/**
* doPost() 方法在post请求的时候调用
* @param req
* @param resp
* @throws ServletException
* @throws IOException
*/
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("HelloServlet2 的doPost方法被调用");
}
}
<servlet>
<servlet-name>HelloServlet2servlet-name>
<servlet-class>com.zanedu.servlet.HelloServlet2servlet-class>
servlet>
<servlet-mapping>
<servlet-name>HelloServlet2servlet-name>
<url-pattern>/hello2url-pattern>
servlet-mapping>
路径:File – Project Structure – Facets – Source Roots – 勾上
package com.zanedu.servlet;
import jakarta.servlet.*;
import jakarta.servlet.http.*;
import java.io.IOException;
public class HelloServlet3 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("HelloServlet3的doGet方法被调用");
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("HelloServlet3的doPost方法被调用");
}
}
ServletConfig类从类名上来看,就知道是Servlet程序的配置信息类
Servlet程序和ServletConfig对象都是由Tomcat负责创建,我们负责调用
Servlet程序默认是第一次访问的时候创建,ServletCongfig是每个Servlet程序创建时,就创建一个对应的ServletConfig对象
package com.zanedu.servlet;
import jakarta.servlet.*;
import jakarta.servlet.http.HttpServletRequest;
import java.io.IOException;
public class HelloServlet implements Servlet {
public HelloServlet() {
System.out.println("1 构造器方法");
}
@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());
}
@Override
public ServletConfig getServletConfig() {
return null;
}
/**
* service 方法是专门用来处理请求和响应的
* @param servletRequest
* @param servletResponse
* @throws ServletException
* @throws IOException
*/
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
System.out.println("3 service === Hello Servlet 被访问了");
//类型转换(因为它有getMethod()方法,而servletRequest是父接口,没有这个方法,因此要转换一下)
HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;
//获取请求的方式
String method = httpServletRequest.getMethod();
// System.out.println(method);
if ("GET".equals(method)) {
doGet();
} else if ("POST".equals(method)) {
doPost();
}
}
/**
* 做get请求的操作
*/
public void doGet() {
System.out.println("get请求");
System.out.println("get请求");
}
/**
* 做post请求的操作
*/
public void doPost() {
System.out.println("post请求");
System.out.println("post请求");
}
@Override
public String getServletInfo() {
return null;
}
@Override
public void destroy() {
System.out.println("4 destroy销毁方法");
}
}
<servlet>
<servlet-name>HelloServletservlet-name>
<servlet-class>com.zanedu.servlet.HelloServletservlet-class>
<init-param>
<param-name>usernameparam-name>
<param-value>rootparam-value>
init-param>
<init-param>
<param-name>urlparam-name>
<param-value>jdbc:mysql://localhost:3306/testparam-value>
init-param>
servlet>
:::tips
每一个ServletConfig所对应的都是自己的HelloServlet,而不能得到别人的信息,因此我们的username和url这里是只放在HelloServlet中,如果我们使用别的Servlet,那么就会访问不到,返回null
:::
:::tips
当我们实现Servlet接口,去重写init方法,是没有问题的。但是当我们是继承HttpServlet类的时候,去重写init方法,必须加上super.init(config);要去调用一下父类的init方法
:::
public class HelloServlet2 extends HttpServlet {
@Override
public void init(ServletConfig config) throws ServletException {
// super.init(config);
System.out.println("重写了init初始化方法,做了一些工作");
}
}
- ServletContext是一个接口,它表示Servlet上下文对象
- 一个web工程,只有一个ServletContext对象实例
- ServletContext对象是一个域对象
- ServletContext是在web工程部署启动的时候创建,在web工程停止的时候销毁
域对象:是可以像Map一样存取数据的对象
这里的域指的是存取数据的操作范围是整个web工程
|
| 存数据 | 取数据 | 删除数据 |
| — | — | — | — |
| Map | put() | get() | remove |
| 域对象 | setAttribute() | getAttribute() | removeAttribute() |
<context-param>
<param-name>usernameparam-name>
<param-value>contextparam-value>
context-param>
<context-param>
<param-name>passwordparam-name>
<param-value>rootparam-value>
context-param>
package com.zanedu.servlet;
import jakarta.servlet.*;
import jakarta.servlet.http.*;
import java.io.IOException;
public class ContextServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//1、获取 web.xml 中配置的上下文参数 context-param
ServletContext servletContext = getServletConfig().getServletContext();
String username = servletContext.getInitParameter("username");
System.out.println("context-param参数username的值是:" + username);
String password = servletContext.getInitParameter("password");
System.out.println("context-param参数username的值是:" + password);
}
}
//2、获取当前的工程路径,格式: /工程路径
System.out.println("当前工程路径:" + servletContext.getContextPath());
package com.zanedu.servlet;
import jakarta.servlet.*;
import jakarta.servlet.http.*;
import java.io.IOException;
public class ContextServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//3、获取工程部署后在服务器硬盘上的绝对路径
/**
* / 斜杠被服务器解析地址为http://ip:port/工程名/ 映射到IDEA代码的web目录
*/
System.out.println("工程部署的路径是:" + servletContext.getRealPath("/"));
System.out.println("工程下imgs目录的绝对路径是:" + servletContext.getRealPath("/imgs"));
System.out.println("工程下imgs目录下1.jpg的绝对路径是:" + servletContext.getRealPath("/imgs/1.jpg"));
}
}
package com.zanedu.servlet;
import jakarta.servlet.*;
import jakarta.servlet.http.*;
import java.io.IOException;
public class ContextServlet1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//获取ServletContext对象
ServletContext servletContext = getServletContext();
// System.out.println(servletContext);
System.out.println("保存之前Context1 中获取域数据key1的值是:" + servletContext.getAttribute("key1"));
servletContext.setAttribute("key1", "value1");
System.out.println("Context1 中获取域数据key1的值是:" + servletContext.getAttribute("key1"));
}
}
:::tips
协议是指双方、或多方,相互约定好的,大家都需要遵守的规则,叫协议
所谓HTTP协议,就是指:客户端和服务端之间通信时,发送的数据,需要遵守的规则,叫HTTP协议
HTTP协议中的数据又叫报文
:::
:::tips
客户端给服务器发送数据叫请求
服务器给客户端回传数据叫响应
:::
:::tips
key : value 组成,不同的键值对,表示不同的含义
:::
:::tips
key : value 不同的请求头,有不同的含义
:::tips
Accept:表示客户端可以接受的数据类型
Accpet-Language:表示客户端可以接受的语言类型
User-Agent:表示客户端浏览器的信号
Host:表示请求时的服务器ip和端口号
:::
:::tips
GET请求有:
POST请求有:
:::tips
key : value 不同的响应头,有其不同的含义
200 | 表示请求成功 |
---|---|
302 | 表示请求重定向 |
404 | 表示请求服务器以及收到了,但是你要的数据不存在(请求地址错误) |
500 | 表示服务器已经收到请求,但是服务器内部错误(代码错误) |
:::tips
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 |
JDEG图形 | .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 |
:::tips
每次只要有请求进入Tomcat服务器,Tomcat服务器就会把请求过来的HTTP协议信息解析好封装到Request对象中,然后传递到service方法或doGet方法或doPost方法中给我们使用。而我们可以通过HttpServletRequest对象,获取到所有请求的信息
:::
getRequestURI() | 获取请求的资源路径 |
---|---|
getRequestURL() | 获取请求的统一资源定位符(绝对路径) |
getRemoteHost() | 获取客户端的ip地址 |
getHeader() | 获取请求头 |
getParameter() | 获取请求的参数 |
getParameterValues() | 获取请求的参数(多个值的时候使用)[如复选框] |
getMethod() | 获取请求的方式GET或POST |
setAttribute(key, value) | 设置域数据 |
getAttribute(key) | 获取域数据 |
getRequestDispatcher() | 获取请求转发对象 |
package com.zanedu.servlet;
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 RequestAPIServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// i. getRequestURI() 获取请求的资源路径
System.out.println("URI => " + req.getRequestURI());
// ii. getRequestURL() 获取请求的统一资源定位符(绝对路径)
System.out.println("URL => " + req.getRequestURL());
// iii. getRemoteHost() 获取客户端的 ip 地址
/*
在IDEA中使用LocalHost访问时,得到的ip地址是:0:0:0:0:0:0:0:1
在IDEA中使用127.0.0.1访问时,得到的ip地址是:127.0.0.1
在IDEA中使用真实ip地址访问时,得到的ip地址是:真实的客户端ip地址
*/
System.out.println("客户端的ip地址:" + req.getRemoteHost());
// iv. getHeader() 获取请求头
//User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.0.0 Safari/537.36
System.out.println("请求头User-Agent:" + req.getHeader("User-Agent"));
// vii. getMethod() 获取请求的方式 GET 或 POST
System.out.println("请求的方式:" + req.getMethod());
}
}
DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Titletitle>
head>
<body>
<form action="http://localhost:8080/07_servlet/param" method="post">
用户名: <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">Java
<input type="checkbox" name="hobby" value="js">JavaScript<br/>
<input type="submit">
form>
body>
html>
package com.zanedu.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.Arrays;
public class ParameterServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("----------doGet----------");
//获取请求的参数
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobby = req.getParameterValues("hobby");
System.out.println("用户名:" + username);
System.out.println("密码:" + password);
System.out.println("兴趣爱好:" + Arrays.asList(hobby));
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//设置请求体的字符集为UTF-8,从而解决post请求的中文乱码问题
//这个要在获取请求参数之前调用才有效
req.setCharacterEncoding("UTF-8");
System.out.println("----------doPost----------");
//获取请求的参数
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobby = req.getParameterValues("hobby");
System.out.println("用户名:" + username);
System.out.println("密码:" + password);
System.out.println("兴趣爱好:" + Arrays.asList(hobby));
}
}
doGet请求的中文乱码解决
doPost请求的中文乱码解决
:::tips
请求转发是指:服务器收到请求后,从一个资源跳转到另一个资源的操作
:::
package com.zanedu.servlet;
import jakarta.servlet.RequestDispatcher;
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 Servlet1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//获取请求的参数(办事的材料)查看
String username = req.getParameter("username");
System.out.println("在Servlet1(柜台1)查看参数(材料):" + username);
//使用域数据,给材料盖一个张并传递到Servlet2(柜台2)去查看
req.setAttribute("key", "柜台1的章");
//问路:Servlet2(柜台2)怎么走
/**
* 请求转发必须要以斜杠打头,/ 斜杠表示地址为http://ip:port/工程名/,映射到IDEA代码的web目录
*/
RequestDispatcher requestDispatcher = req.getRequestDispatcher("/servlet2");
//走向Servlet2(柜台2)
requestDispatcher.forward(req, resp);
}
}
package com.zanedu.servlet;
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 Servlet2 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//获取请求的参数(办事的材料)查看
String username = req.getParameter("username");
System.out.println("在Servlet2(柜台2)查看参数(材料):" + username);
//查看柜台1是否有盖章
Object key = req.getAttribute("key");
System.out.println("柜台1是否有盖章:" + key);
//处理自己的业务
System.out.println("Servlet2 处理自己的业务");
}
}
RequestDispatcher requestDispatcher = req.getRequestDispatcher("/WEB-INF/form.html");
RequestDispatcher requestDispatcher = req.getRequestDispatcher("http://www.baidu.com");
DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Titletitle>
<base href="http://localhost:8080/07_servlet/a/b/">
head>
<body>
这是a下的b下的c.html页面 <br/>
<a href="../../index.jsp">跳回首页a>
body>
html>
即永远都是
"http://localhost:8080/07_servlet/a/b/../../index.jsp"
特殊情况:response.sendRediect(“/”); 把斜杠发送给浏览器解析,得到http://ip:port/
:::
:::tips
HttpServletResponse类和HttpServletRequest类一样,每次请求进来,Tomcat服务器都会创建一个Response对象传递给Servlet程序去使用。HttpServletRequest表示请求过来的信息,HttpServletResponse表示所有响应的信息
我们如果需要设置返回给客户端的信息,都可以通过HttpServletResponse对象来进行设置
:::
字节流 | getOutputStream() | 常用于下载(传递二进制数据) |
---|---|---|
字符流 | getWriter() | 常用于回传字符串(常用) |
package com.zanedu.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
public class ResponseIOServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.getOutputStream();
resp.getWriter();
}
}
package com.zanedu.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
public class ResponseIOServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//要求:往客户端回传 字符串 数据
PrintWriter writer = resp.getWriter();
//可以用write,也可以用println等等
writer.write("response`s content!!!");
}
}
package com.zanedu.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
public class ResponseIOServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//设置服务器字符集为UTF-8
resp.setCharacterEncoding("UTF-8");
//通过响应体设置浏览器也使用UTF-8字符集
resp.setHeader("Content-type", "text/html; charset=UTF-8");
//要求:往客户端回传 字符串 数据
PrintWriter writer = resp.getWriter();
writer.write("我帅吗");
}
}
//它会同时设置服务器和客户端都使用UTF-8字符集,还设置了响应头
//注意点:此方法一定要在获取流对象之前调用才有效
resp.setContentType("text/html; charset=UTF-8");
:::tips
请求重定向:是指客户端给服务器发请求,然后服务器告诉客户端说,我给你一些地址,你去新地址访问,叫请求重定向(因为之前的地址可能已经被废弃)
:::
package com.zanedu.servlet;
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 Response1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("曾到此一游 Response1");
req.setAttribute("key1", "value1");
//设置响应状态码 302,表示重定向(已搬迁)
resp.setStatus(302);
//设置响应头,说明新的地址在哪里
// resp.setHeader("Location", "http://localhost:8080/07_servlet/response2");
}
}
package com.zanedu.servlet;
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 Response2 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println(req.getAttribute("key1"));
resp.getWriter().write("response2 到此一游");
}
}
:::tips
不能访问到WEB-INF目录下的文件,是因为WEB-INF是受保护的,而浏览器是不能够访问受保护的,当重新第二次到新地址,也是浏览器发送请求,因此不能访问到受保护的WEB-INF目录
:::
resp.setHeader("Location", "http://localhost:8080/07_servlet/WEB_INF/form.html");
resp.setHeader("Location", "http://www.baidu.com");