Web 服务器-Tomcat
- Web服务器是一个应用程序(软件),对HTTP协议的操作进行封装,使得程序员不必直接对协议进行操作,让Web开发更加便捷,主要功能是“提供网上信息浏览服务”
Tomcat
- 概念:Tomcat是Apacge软件基金会一个核心项目,是一个开源免费的轻量级Web服务器,支持Servlet/JSP 少量JavaEE规范
- JavaEE:Java Enterprise Edition,Java企业版,值Java企业级开发的技术规范总共,包含13项技术规范:JDBC、JNDI、EJB、RMI、JSP、Servlet、XML、JMS、Java IDL、JTS、JTA、JavaMail、JAF
- Tomcat也被称为Web容器、Servlet容器,Servlet需要依赖于Tomcat才能运行
Servlet
- Servlet是Java提供的一门动态web资源开发技术
- Servlet是JavaEE规范之一,起始就是一个接口,将来我们需要定义Servlet类实现Servlet接口,并由web服务器运行Servlet
Servlet快速入门
- 创建web项目,导入Servlet依赖坐标
<dependency>
<groupld>javax.servlet</groupld>
<artifacttld>javax.servlet-api</artifacttld>
<version>3.1.0</version>
<scope>provided</scope>
</dependency>
- 创建:定义一个类,实现Servlet接口,并重写接口中所有方法,并在service方法中输入一句话
public class ServeletDemo1 implements Servlet {
public void service(){}
}
- 配置:在类上使用@WebServlet注解,配置该Servlet的访问路径
@WebServlet("/demo1")
public class ServletDemo1 implements Servlet {
- 访问:启动Tomcat,浏览器输入URL访问该Servlet
http://localhost:8080/web-demo/demo1
Servlet执行流程
- Servlet由谁创建?Servlet方法由谁调用?
Servlet由web服务器创建,Servlet方法由web服务器调用
- 服务器怎么知道Servlet中一定有Servlet方法?
因为我们自定义的Servlet,必须实现Servlet接口并复写其方法,而Servlet接口中有Servlet方法
Servlet生命周期
- 对象的声明周期指一个对象从被创建到被销毁的整个过程
- Servlet运行在Servlet容器(web服务器)中,其生命周期由容器来管理,分为4个阶段
-
加载和实例化:默认情况下,当Servlet第一次被访问时,由容器创建Servlet对象
- 不默认的情况下,可以改变Servlet的创建时机,在配置路径时,可以通过配置参数来改变创建时机
@WebServlet(urlPatterns = "/demo", loadOnStartup = 1)- 负整数:第一次被访问时创建Servlet对象
- 0或正整数:服务器启动时创建Servlet对象,数字越小优先级越高
-
初始化:在Servlet实例化之后,容器将调用Servlet的init() 方法初始化这个对象,完成一些如加载配置文件、创建连接等初始化的工作,给方法只调用一次
-
请求处理:每次请求Servlet时,Servlet容器都会调用一次Servlet的service() 方法对请求进行处理
-
服务中止:当需要释放内存或者容器关闭时,容器就会调用Servlet实例的destroy() 方法完成资源的释放,在destroy()方法调用后,容器会释放这个Servlet实例,该实例随后会被Java的垃圾收集器所回收
Servlet方法介绍
- 初始化方法,在Servlet被创建时执行,只执行一次
void init(ServletConfig config)
- 提供服务方法,每次Servlet被访问,都会调用方法
void Servlet(ServletRequest req, ServletResponse res)
- 销毁方法,当Servlet被销毁时,调用该方法,在内存释放或服务器关闭时销毁Servlet
void destroy()
- 获取ServletConfig对象
ServletConfig getServetConfig()
- 获取Servlet信息
String getServletInfo()
Servlet体系结构
我们将来开发B/S架构的web项目,都是针对HTTP协议,所以我们自定义Servlet,会继承HttpServlet
-
HttpServlet使用步骤
- 继承HttpServlet
- 重写doGet和doPost方法
-
HttpServlet原理
- 获取请求原理,并根据不同的请求方式,调用不同的doXxx方法
urlPattern配置
Servlet要想被访问,必须配置其访问路径(urlPattern)
- 一个Servlet,可以配置多个urlPattern
@WebServlet(urlPatterns = {"/demo1", "demo2"})
-
urlPattern配置规则
-
精确匹配
- 配置路径:
@WebServlet("/user/select") - 访问路径:
localhost:8080/web-demo/user/select
- 配置路径:
-
目录匹配
-
配置路径:
@WebServlet("/user/*") -
访问路径:
localhost:8080/web-demo/user/aaalocalhost:8080/web-demo/user/bbb
-
-
扩展名匹配
-
配置路径:
@WebServlet("*.do") -
访问路径:
localhost:8080/web-demo/aaa.dolocalhost:8080/web-demo/bbb.do
-
-
任意匹配
-
配置路径:
@WebServlet("/")@WebServlet("/*") -
访问路径:
localhost:8080/web-demo/hehelocalhost:8080/web-demo/haha -
/ 和 /* 区别:
- 当我们的项目中的Servlet配置了 “ / ” ,会覆盖掉tomcat中的DefaultServlet,当其他的 url-pattern 都匹配不上时都会走这个Servlet
- 当我们的项目中配置了 “ /* ” ,意味着匹配任意访问路径
-
-
优先级:精确路径 > 目录路径 > 扩展名路径 > /* > /
-
XML配置方式编写Servlet
-
Servlet从3.0版本后开始支持使用注解配置,3.0版本前只支持XML配置我文件的配置方式
-
步骤:
- 编写Servlet类
- 在web.xml中配置该Servlet
<servlet> <servlet-name>demo5</servlet-name> <servlet-class>com.itheima.web.servlat.ServletDemo5</servlet-class> </servlet> <servlet-mapping> <servlet-name>demo5</servlet-name> <url-pattern>/demo5</url-pattern> </servlet-mapping>
Request(请求)&Response(响应)
- Request:获取请求数据
- Response:设置响应数据
Request
Request继承体系
- Tomcat需要解析请求数据,封装为request对象,所以就需要定义这个实现类,并且创建request对象传递到service方法中
- 使用request对象,查阅JavaEE API文档的HttpServletRequest接口
Request获取请求数据
获取请求数据
- 请求数据分为3部分:
-
请求行:
GET/request-demo/req1?usernames=zhansan HTTP/1.1- String getMethod():获取请求方式:GET
- String getContextPath():(动态)获取虚拟目录(项目访问路径):/request-demo
- StringBuffer getRequestURL():获取URL(统一资源定位符):http://localhost:8080/request-demo/req1
- String getRequestURI():获取URI(统一资源标识符):/request-demo/req1
- String getQueryString():获取请求参数(GET方式):usernames=zhansan&password=123
-
请求头:
User-Agent:Mozilla/5.0 Chrome/91.0.4472.106- String getHeader(String name):根据请求头名称,获取值
-
请求体:
username=superbaby&password=123- ServletInputStream getInputStream():获取字节输入流
- BufferedReader getReader():获取字符输入流
通用方式获取请求参数
-
请求参数获取方式:
- GET方式:
String getQueryString() - POST方式:
BufferedReader getReader()
- GET方式:
GET请求方式和POST请求方式区别主要在于获取请求参数的方式不一样,是否可以提供一种统一获取请求参数的方式,从而统一doGet和doPost方法内的代码
//获取请求方式
String method = this.getMethod();
//判断
if ("GET".equals(method)) {
//GET方式获取请求参数
params = this.getQueryString();
} else if ("POST".equals(method)) {
//POST方式获取请求参数
BufferedReader reader = this.getReader();
params = reader.readLine();
}
Map<String, String[]> getParameterMap():获取所有参数Map集合String[] getParametreValues(String name):根据名称获取参数值(数组)String getParameter(String name):根据名称获取参数值(单个值)
- 使用通用方式获取请求参数后,屏蔽了GET和POST的请求方式代码的不同,则代码可以定义为如下格式:
@WebServlett("/reqDemo3")
public class RequestDemo3 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) {
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) {
this.doGet(req, resp);
}
}
- 可以通过Servlet模板创建Servlet更高效
请求参数中文乱码处理
-
请求参数如果存在中文数据,则会乱码
-
解决方案:
- POST:设置输入流的编码
req.setCharacterEncoding("UTF-8");-
GET:原理:编解码方式不一样
解决方案:
- 先对乱码数据进行编码:转为字节数组
byte [] bytes = username.getBytes(StandardCharsets.ISO_8859_1);- 字节数组解码
username = new String(bytes, StandardCharsets.UTF_8);- 最终方案(GET 和 POST 通用):先解码,再编码
new String(username.getBytes("ISO-8859-1"), "UTF-8"); -
URL编码
-
将字符串按照编码方式转为二进制
-
每个字节转为2个16进制数并在前面加上%
- 编码:
URLEncoder.encoder(str, "utf-8");- 解码:
URLEncoder.decodes(s, "ISO-8859-1");
-
Tomcat8.0 之后,已将GET请求乱码问题解决,设置默认的解码方式为UTF-8
Request请求转发
-
请求转发(forward):一种在服务器内部的资源跳转方式
-
实现方式:
req.getRequestDispatcher("资源B路径").forward(req.resp);
-
请求转发资源间共享数据:使用Request对象
void setAttribute(String name, Object o):存储数据到request域中Object getAttribute(String name):根据key,获取值void removeAttribute(String name):根据key,删除键值对
-
请求转发特点
- 浏览器地址栏路径不发生变化
- 只能转发到当前服务器的内部资源
- 一次请求,可以在转发的资源间使用request共享数据
Response
Response继承体系
Response设置响应数据功能介绍
- 响应数据分为3部分:
-
响应行:
HTTP/1.1 200 OKvoid setStatus(int sc):设置响应状态码
-
响应头:
Content-Type:text/htmlvoid setHeader(String name, String value):设置响应头键值对
-
响应体:
<html><head></head><body></body></html>PrintWriter gerWriter():获取字符输出流ServletOutputStream getOutputStream():获取字节输出流
Response完成重定向
-
重定向(Redirect):一种资源跳转方式
-
实现方式:
resp.setStatus(302);
resp.setHeader("location", "资源B的路径")
- 简化方式实现重定向
resp.sendRedirect("资源B的路径")
-
重定向特点:
- 浏览器地址栏路径发生变化
- 可以重定向到任意位置的资源(服务器内部、外部皆可)
- 两次请求,不能在多个资源使用request共享数据
路径问题
-
明确路径谁使用?
- 浏览器使用:需要加虚拟目录(项目访问路径)
- 服务端使用:不需要虚拟目录
-
例:
<a href = '路径'>:加虚拟路径<form action = '路径'>:加虚拟路径`req.getRequestDispatcher("路径"):不加虚拟路径resp.sendRedirect("路径"):加虚拟路径
-
动态获取虚拟路径(让代码的耦合性更低,利于维护)
//动态获取虚拟目录
String contextPath = request.getContextPath();
response.sendRedirect(contextPath + "资源路径")
Response响应字符数据
- 响应字符数据:设置字符数据的响应体
- 使用:
- 设置响应的数据格式和字符集
response.setContentType("text/html; charset=UTF-8");
- 通过Response对象获取字符输出流
PrintWriter writer = resp.getWriter();
- 写数据
writer.writer("aaa");
细节:
- 该流不需要关闭,随着响应结束,response对象销毁,由服务器关闭
- 中文数据会乱码,原因通过Response获取的字符输出流默认编码:ISO-8859-1
Response响应字节数据
- 使用:
- 通过Response对象获取字符输出流
ServletOutputStream outputStream = resp.getOutputStream();
- 写数据
outputStream.write(字节数据);
- IOUtils工具类使用
- 导入坐标
<dependency>
<groupld>commons-io</groupld>
<artifactld>commons-io</artifactld>
<version>2.6</version>
</dependency>
- 使用
IOUtils.copy(输入流,输出流);