JavaWeb技术栈:
(1)B/S架构:Browser/Server,浏览器/服务器架构模式,它的特点是,客户端只需要浏览器,应用程序的逻辑和数据都存储在服务器端。浏览器只需要请求服务器,获取Web资源,服务器把Web资源发送给浏览器即可。
	好处:易于维护升级:服务器端升级后,客户端无需任何部署就可以使用到新的版本。
(2)静态资源:HTML、CSS、JavaScript、图片等。负责页面展现。
(3)动态资源:Servlet、JSP等。负责逻辑处理。
(4)数据库:负责存储数据。
(5)HTTP协议:定义通信规则。
(6)Web服务器:负责解析HTTP协议,解析请求数据,并发送响应数据。
HTTP:超文本传输协议,规定了浏览器和服务器之间数据传输的规则。
特点:
	(1)基于TCP协议:面向连接,安全。
	(2)基于请求-响应模型的:一次请求对应一次响应。
	(3)HTTP协议是无状态的协议:对于事务处理没有记忆能力。每次请求-响应都是独立的。
		3.1)缺点:多次请求 间不能共享数据。(Java中使用会话技术(Cookie、Session)来解决这个问题)
		3.2)优点:速度快。
HTTP-请求数据格式:
	(1)请求数据分为3部分:
		1.1)请求行:请求数据的第一行。其中GET表示请求方式,/表示请求资源路径,HTTP/1.1表示协议版本。
		1.2)请求头:第二行开始,格式为key: value形式。
		1.3)请求体:POST请求的最后一部分,存放请求参数
	(2)常见的HTTP请求头:
		2.1)Host:表示请求的主机名
		2.2)User-Agent:浏览器版本
		2.3)Accept:表示浏览器能接收的资源类型,如text/*,image/*或者*/*表示所有
		2.4)Accept-Language:表示浏览器偏好的语言,服务器可以据此返回不同语言的网页
		2.5)Accept-Encoding:表示浏览器可以支持的压缩类型,例如gzip,deflate等
	(3)GET请求和POST请求区别:
		3.1)GET请求请求参数在请求行中,没有请求体。POST请求请求参数在请求体中。
		3.2)GET请求请求参数大小有限制,POST没有。
HTTP-响应数据格式:
	(1)响应数据分为3部分:
		1.1)响应行:响应数据的第一行。其中HTTP/1.1表示协议版本,200表示响应状态码,OK表示状态码描述。
		1.2)响应头:第二行开始,格式为key:value形式
		1.3)响应体:最后一部分。存放响应数据。
	(2)常见的HTTP响应头:
		2.1)Content-Type:表示该响应内容的类型,例如text/html,image/jpeg
		2.2)Content-Length:表示该响应内容的长度(字节数)
		2.3)Content-Encoding:表示该响应压缩算法,例如gzip
		2.4)Cache-Control:指示客户端应任何缓存,例如max-age=300:表示可以最多缓存300秒
	(3)状态码大全:https://cloud.tencent.com/developer/chapter/13553
Web服务器:是一个应用程序(软件),对HTTP协议的操作进行封装,使得程序员不必直接对协议进行操作,让Web开发更加便捷。主要功能是“提供网上信息浏览服务”。
Servlet:是Java提供的一门动态web资源开发技术
(1)Servlet快速入门:
	1.1)创建web项目,导入Servlet依赖坐标
		<dependencies>
			<dependency>
				<groupId>javax.servlet</groupId>
				<artifactId>javax.servlet-api</artifactId>
				<version>3.1.0</version>
				<scope>provided</scope>
			</dependency>
		</dependencies>
	1.2)创建:定义一个类,实现Servlet接口,并重写接口中所有方法,并在servlet方法中输入一句话
		public class ServletDemo1 implements Servlet {}
	1.3)配置:在类上使用@WebServlet注解,配置该Servlet的访问路径
		@WebServlet("/demo1")
		public class ServletDemo1 implements Servlet {
	1.4)访问:启动Tomcat,浏览器输入URL访问该Servlet
(2)Servlet执行流程:
	2.1)Servlet由谁创建?Servlet方法由谁调用?
		Servlet由Web服务器创建,Servlet方法由Web服务器调用。
	2.2)服务器怎么知道Servlet中一定有service方法?
		因为我们自定义的Servlet,必须实现Servlet接口并复写其方法,而Servlet接口中有service方法
(3)HttpServlet使用步骤:
	3.1)继承HttpServlet
	3.2)重新doGet和doPost方法
(4)HttpServlet原理:
	获取请求方式,并根据不同的请求方式,调用不同的doXxx方法
(5)Servlet urlPattern配置
	5.1)Servlet想要被访问,必须配置其访问路径(urlPattern)
		5.1.1)一个Servlet,可以配置多个urlPattern
			@WebServlet(urlPatterns = {"/demo1","/demo2"})
		5.1.2)urlPattern配置规则
			<1>精确匹配
			<2>目录匹配
			<3>扩展名匹配
			<4>任意匹配
Requet(请求) & Response(响应)
(1)Request:获取请求数据
	1.1)请求行:GET/request-demo/req1?username=zhangsan 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方式):username=zhangsan&password=123
	1.2)请求头:User-Agent:Mozilla/5.0 Chrome/91.0.4472.106
		String getHeader(String name):根据请求头名称,获取值
	1.3)请求体:username=superbaby&password=123
		ServletInputStream getInputStream():获取字节输入流
		BufferedReader getReader():获取字符输入流
(2)Response:设置响应数据
	2.1)响应行:HTTP/1.1 200 OK
		void setStatus(int sc):设置响应状态码
	2.2)响应头:Content-Type:text/html
		void setHeader(String name, String value):设置响应头键值对
	2.3)响应体:<html><head></head><body></body></html>
		PrintWriter getWriter():获取字符输出流
		ServletOutputStream getOutputStream():获取字节输出流
(3)Request通用方式获取请求参数
	3.1)GET方式:
		String getQueryString()
	3.2)POST方式:
		BufferedReader getReader()
	3.3)通用获取请求参数方法:
		Map<String,String[]>getParameterMap():获取所有参数Map集合
		String[] getParameterValues(String name):根据名称获取参数值(数组)
		String getParameter(String name):根据名称获取参数值(单个值)
(4)Request请求参数中文乱码处理
	4.1)解决方案:
		POST:设置输入流的编码
			req.setCharacterEncoding("UTF-8");
		GET:(也可以解决POST方式的乱码)
			乱码原因:tomcat进行URL解码,默认的字符集ISO-8859-1
			解决方法:
				<1>先对乱码数组进行编码:转为字节数组
					byte[] bytes = username.getBytes(StandarCharsets.ISO_8859_1);
				<2>字节数组解码
					username = new String(bytes, StandardCharsets.UTF_8)
	4.2)URL编码:
		URLEncoder.encode(str, "utf-8");
		4.2.1)将字符串按照编码方式转为二进制
		4.2.2)每个字节转为2个16进制数并在前边加上%
	4.3)URL解码:
		URLDecoder.decode(s,"ISO-8859-1");
(5)请求转发(forward):一种在服务器内部的资源跳转方式
	5.1)实现方式:req.getRequestDispatcher("资源B路径").forward(req,resp);
	5.2)请求转发资源间共享数据:使用Request对象
		<1> void setAttribute(String name, Object o):存储数据到request域中
		<2> Object getAttribute(String name):根据key,获取值
		<3> void removeAttribute(String name):根据key,删除该键值对
	5.3)请求转发特点:
		<1> 浏览器地址栏路径不发生变化
		<2> 只能转发到当前服务器的内部资源
		<3>一次请求,可以在转发的资源间使用request共享数据
(6)Response完成重定向:
	6.1)重定向(Redirect):一种资源跳转方式
	6.2)实现方式:
		resp.setStatus(302);
		resp.setHeader("location","资源B的路径");
		简化方式:resp.sendRedirect("资源B的路径");
	6.3)重定向特点:
		<1> 浏览器地址栏路径发生变化
		<2> 可以重定向到任意位置的资源(服务器内部、外部均可)
		<3> 两次请求,不能在多个资源使用request共享数据
(7)明确路径谁使用?
	7.1)浏览器使用:需要加虚拟目录(项目访问路径)
	7.2)服务端使用:不需要加虚拟目录
(8)Response响应字符数据:
	8.1)使用:
		<1> 通过Response对象获取字符输出流
			PrintWriter writer = resp.getWriter();
		<2> 写数据
			writer.write("aaa");
	8.2)注意:
		<1> 该流不需要关闭,随着响应结束,response对象销毁,由服务器关闭
		<2> 中文数据乱码:原因通过Response获取的字符输出流默认编码:ISO-8859-1
			resp.setContentType("text/html;charset=utf-8");
(9)Response响应字节数据:
	9.1)使用:
		<1> 通过Response对象获取字符输出流
			ServletOutputStream outputStream = resp.getOutputStream();
		<2> 写数据
			outputStream.write(字节数据);
	9.2)IOUtils工具类使用
		<1> 导入坐标
			<dependency>
				<groupId>commons-io</groupId>
				<artifactId>commons-io</artifactId>
				<version>2.6</version>
			</dependency>
		<2> 使用
			IOUtils.copy(输入流,输出流);