一、Servlet技术
Servlet是Java接口规范之一
Servlet是JavaWeb三大组件之一,三大(Servlet程序、Filter过滤器、Listener监听器)
Servlet是运行在服务器的java小程序,可以接收客户端发送的请求,并响应给客户端
(1)IDEA导包:
- 将 tomcat/lib文件下的 servlet-api.jar ,复制到工程web>WEB-INF>lib文件下
- 右键 servlet-api.jar > as lib ,添加依赖到工程里。
或者
- Project Settings --> libraries ---> 左边束栏“+” -->java (选择你要添加的jar包)
- Project Settings --> Modules -->选择module--> SDK附件的“ + ” --> library (刚添加的jar包)
- Project Settings -->Artifacts -->选择module ---> 右下角有 fix --> add
(2)Servlet初始化:
- 在moulde(自定义) > src >packed(自定义) > 创建java类 实现 Servlet接口
- :其中有 service方法是专门用来处理请求和响应的,sout("已被访问")
- web > WEB-INF > web.xml
-
<?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.Servelet.HelloServlet</servlet-class> </servlet> <!-- servlet-mapping 给Servlet程序配置访问地址--> <servlet-mapping> <!-- servlet-name 告诉Tomcat服务器,当前配置的访问地址给哪个Servlet程序使用--> <servlet-name>HelloServlet</servlet-name> <!-- url-pattern 配置访问地址 / : 表示http://ip:port/工程路径/ 工程路径:SerTest_war_exploded /hello : 表示http://ip:port/工程路径/hello --> <url-pattern>/hello</url-pattern> </servlet-mapping> </web-app>
-
(3.) Servlet访问原理
(4.) Servlet生命周期
- 执行构造器方法
- 只有第一次访问调用
- 执行init初始化方法
- 只有第一次访问调用
- 执行service方法
- 每次访问都会调用
- 执行destroy方法
- web工程停止时候 (tomcat服务器中止)调用
(5) IDEA 创建 Servlet方法
- packed(自定义) > 右键 new Servlet >
(6)Servlet类的继承体系
2.ServletConfig类
servlet的配置信息类
servlet程序和ServletConfig对象都是由Tomcat负责创建,我们负责使用
servlet程序默认是第一次访问的时候创建,ServletConfig是每个servlet程序创建时,就创建一个对应的ServletConfig对象
(在继承HttpServlet类的子类中的 doGet\doPost方法里可以调用getServletConfig() 返回对象)
(1)ServletConfig类的三大作用
- 获取Servlet程序的别名 servlet-name的值
- 获取init初始化方法的参数 init-param
- 获取ServletContext对象
web\web.xml
<servlet>
<servlet-name>HelloServlet</servlet-name>
<servlet-class>com.Servelet.HelloServlet</servlet-class>
<!-- init-param是初始化参数-->
<init-param>
<!-- param-name是参数名-->
<param-name>username</param-name>
<!-- param-name是参数值-->
<param-value>root</param-value>
</init-param>
</servlet>
使用
@Override
public void init(ServletConfig servletConfig) throws ServletException {
System.out.println("2.init初始化方法");
System.out.println("程序名:"+servletConfig.getServletName());
System.out.println("初始化参数"+servletConfig.getInitParameter("username"));
System.out.println("getServletContext对象:"+servletConfig.getServletContext());
}
3.ServletContext类
(1)介绍:
- ServletContext是一个接口,它表示Servlet上下文对象
- 一个web工程,只有一个ServletContext对象实例
- ServletContext 在web工程部署启动的时候创建,在web工程停止时候销毁
- ServletContext对象是一个域对象
- 域对象:像Map一样存取数据的对象 ,域指的是存取数据的操作范围
(2)作用:
- 获取web.xml中配置的上下文参数 context-param
- 获取当前的工程路径,格式:/工程路径
- 获取工程部署后在服务器硬盘上的绝对路径
- 像Map一样存取数据
<?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">
<!-- context-param是上下文参数(属于整个web工程 共用的)-->
<context-param>
<param-name>username</param-name>
<param-value>admin</param-value>
</context-param>
<context-param>
<param-name>password</param-name>
<param-value>123456</param-value>
</context-param>
</web-app>
某路由接口:
public class ContextServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 获取web.xml中配置的上下文参数 context-param
ServletContext context = getServletConfig().getServletContext();
String username = context.getInitParameter("username");
String password = context.getInitParameter("password");
System.out.println(username+":"+password);
// 获取当前的工程路径,格式:/工程路径
System.out.println("当前工程路径:"+context.getContextPath()); // /SerTest_war_exploded
// 获取工程部署后在服务器硬盘上的绝对路径
// 斜杠'/'被服务器解析地址为: http://ip:port/工程名/ 映射到IDEA代码的web目录
// http://ip:port/工程名/ ==> (tomcat --->整个IDEA项目out目录下工程文件(复制我们写的) ---> 我们写的工程文件下web目录) --> out目录
System.out.println("工程部署的绝对路径:"+context.getRealPath("/"));// G:\IDEA_Work\JavaWeb\out\artifacts\SerTest_war_exploded\
System.out.println("工程下css目录的绝对路径"+context.getRealPath("/css"));
System.out.println("工程下imgs目录的1.png绝对路径"+context.getRealPath("/css/1.png"));
}
4.Http协议
客户端和服务器之间通信时,发生的数据,需要遵守的规则
Http协议中的数据又叫报文,
(2)请求Http协议格式
- GET请求
- 请求行
- 请求的方式 GET
- 请求的资源路径+?+请求参数
- 请求的协议的版本号 HTTP/1.1
- 请求头
- key:value组成 不同的键值对,表示不同含义
- 请求行
- POST请求
- 请求行
- 请求的方式 POST
- 请求的资源路径+?+请求参数
- 请求的协议的版本号 HTTP/1.1
- 请求头
- key:value组成 不同的键值对,表示不同含义
- 空行
- 请求体 (发生给服务器的数据)
- 请求行
- 常用请求头
- Accept: 表示客户端可接收的数据类型
- Accept-Language: 表示客户端可接收的语言类型
- User-Agent: 表示客户端浏览器的信息
- Host: 表示请求时 服务器的ip和端口号
- 如何区分GET、POST请求
- GET请求有:
- form标签 method=get
- a标签
- link标签引入css
- Script标签引入js
- img标签引入图片
- iframe引入html页面
- 在浏览器地址中输入地址 回车
- POST请求
- form标签 method=POST
- GET请求有:
(3)响应的HTTP协议格式
- 响应的HTTP协议格式
- 响应行
- 响应的协议和版本号
- 响应状态码
- 响应状态描述符
- 响应头
- key:value 不同响应头,有不同含义
- 空行
- 响应体 (服务器回传给客户端的数据)如下:回传了 html文本信息
-
常见的响应码说明
- 200 表示请求成功
- 302 表示请求重定向
- 404 表示请求服务器已经收到,但你要的数据不存在(请求地址不存在)
- 500 表示服务器已经收到请求,但是服务器内部错误(代码错误)
- 响应行
(4)MIME类型说明
5.HttpServletRequest类
(1)作用
每次只要有请求进入Tomcat服务器,Tomcat服务器就会把请求过来的HTTP协议信息解析好封装到Request对象中。然后传递到service方法(doGet和doPost)中给我们使用。我们可以通过HttpServletRequest对象,获取到所有请求的信息。
(2)HttpServletRequest类的常用方法
- getRequestURI( ) 获取请求的资源路径
- getRequestURL() 获取请求的统一资源定位符(绝对路径)
- getRemoteHost() 获取客户端的ip地址
- getHeader( ) 获取请求头
- getParameter() 获取请求的参数
- getParameterValues() 获取请求的多个参数
- getMethod() 获取请求的方式(GET\POST)
- setAttribute(key,value) 设置域数据
- getAttribute(key) 获取域数据
- getRequestDispatcher() 获取请求转发对象
-
public class RequestAPIServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // getRequestURI( ) 获取请求的资源路径 System.out.println("URI:"+req.getRequestURI()); // /ServletTest2_war_exploded/ras // getRequestURL() 获取请求的统一资源定位符(绝对路径) System.out.println("URL:"+req.getRequestURL()); // http://localhost:8080/ServletTest2_war_exploded/ras // getRemoteHost()获取客户端的ip地址 System.out.println("客户Id地址:"+req.getRemoteHost()); // 0:0:0:0:0:0:0:1 IPV6 // getHeader( ) 获取请求头 System.out.println("请求头User-Agent: "+req.getHeader("User-Agent")); //Mozilla/5.0 (Windows NT 10.0; Win64; x64)... // getMethod()获取请求的方式(GET\POST) System.out.println("请求的方式: "+req.getMethod()); // GET } @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+" 密码:"+password+" 兴趣:"+ Arrays.asList(hobby));//用户名:13267126427 密码:123123 兴趣:[cpp, C, java] } }
(3)Post中文乱码
//需要请求体字符集为UTF-8 必须要在获取请求参数前调用才会生效
req.setCharacterEncoding("UTF-8");
(4)请求的转发
服务器收到请求后,从一次资源跳转到另一个资源的操作叫请求转发
请求转发的特点
- 浏览器地址栏没有变化 (只调整到 /Servlet1 ,没有去/Servlet2 )
- 他们是一次请求 (对客户端和服务器来说 一次进 一次出)
- 他们共享Request域中的数据
- 在路由中 可以通过请求转发到web\WEB-INF目录下 (在浏览器中访问不到的)
- 不可以访问工程以外的资源
public class Servlet1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String username = request.getParameter("username");
System.out.println("在柜台1查看材料"+username);
//设置域值,让另一个servlet也能够使用该参数
request.setAttribute("key","柜台1的章");
//请求转发必须以”/"打头,斜杠表示地址为: http://ip:端口号/工程名/ 映射到IDEA代码的web目录
//("/WEB-INF/form.html") 单纯表示可以访问web-inf的文件
RequestDispatcher requestDispatcher = request.getRequestDispatcher("/servlet2");
requestDispatcher.forward(request,response);//转向servlet2
}
}
(5)Web中的相对路径、绝对路径
相对路径:
。 :表示当前路径
。。 :表示上一级目录
资源名 :表示当前目录/资源名
绝对路径:
http://localhost:8080/ServletTest2_war_exploded/index.html
(1)Base标签(相对路径参考)
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<!-- base标签设置页面相对路径工作时参照的路径 href是 参考的地址值-->
<base href="http://localhost:8080/ServletTest2_war_exploded/a/b/c.html">
</head>
<body>
<p>这是web\a\b\c.html</p>
<a href="../../index.html">跳回首页</a>
</body>
</html>
(2)web中“/”斜杠的不同意义
在web中 / 是一种绝对路径
- ” / “ 如果被浏览器解析,得到的地址是:http://ip:port/
-
<a href="/">斜杠</a> -------------↑
-
- “ / ” 如果被服务器解析,得到的地址是:http://ip:port/工程路径
-
<url-pattern>/servlet1</url-pattern> -----↑
- servletContext.getRealPath("/") -------------------↑
- request.getRequestDispatcher("/") ----------------↑
- 特殊情况:response.senRediect("/"); 斜杠发送给浏览器解析 得http://ip:port/
-
6.HttpServletResponse类的作用
同HttpServletRequest一样,每次请求进来,tomcat服务器都会创建一个Response对象,传递给Servlet程序使用,HttpServletRequest表示请求消息,HttpServletResponse表示响应消息
(1)两个响应流的说明
- 字节流 getOutputSream(); 常用于下载(传递二进制)
- 字符流 getWriter(); 常用于回传字符串
(2)统一设置服务器和浏览器的字符集
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//同时设置服务器和客户端使用UTF-8字符集,还设置了响应头
//此方法一定要在获取流对象之前调用 才有效
response.setContentType("text/html;charset=UTF-8");
System.out.println(response.getCharacterEncoding());//UTF-8
//获取流对象
PrintWriter writer = response.getWriter();
writer.write("我是中文");
}
(3)请求重定向
客户端给服务器发请求,然后服务器告诉客户端说,我给你新地址,你去新地址访问
1.请求重定向的特点
- 浏览器地址栏会发生变化
- 两次请求
- 不共享Request域数据
- 不能访问WEB-INF文件下的资源
- 能访问工程以外的资源(百度、新浪)
重定向方式1
"浏览器直接访问Response1的路由,会自动跳转到Response2的路由"
public class Response1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("到此一游 Response1");
//设置响应码302 表示重定向(已搬迁)
response.setStatus(302);
//设置响应头,并说明新地址
response.setHeader("Location","http://localhost:8080/SerTest_war_exploded/res2");
}
重定向方式2
response.sendRedirect("http://localhost:8080/SerTest_war_exploded/res2");