javaweb学习笔记13

136 阅读10分钟

Servlet & HTTP协议 & Request

Servlet的体系结构:
Servlet  --->接口
    | 
GennerricServlet --->抽象类
    |
HttpServlet  --->抽象类
    *GenernericServlet:将Servlet接口中的其他方法做了空实现,只有service方法是抽象方法
        *将来定义Servlet实现类时,可以继承GennericServlet,实现service方法即可
    *HTTPServlet:对HTTP协议的一种封装,简化操作
        1.继承HttpServlet
        2.重写doGet/doPost方法
Servlet的相关配置
1.urlpartten:servlet访问路径
    1.一个servlet可以定多个访问路径(数组):@WebServlet({"/d4","/dd4","/ddd4"})
    2.路径定义规则:
        1./xxx 路径匹配
        2./xxx/xxx...多层路径,目录结构
            /*:任意的路径都可以访问
        3.*.do:扩展名匹配,任意的路径名加后缀(.do)即可访问,后缀可以任意
HTTP:
* 概念:Hyper Text Transfer Protocol 超文本传输协议
	* 传输协议:定义了,客户端和服务器端通信时,发送数据的格式
	* 特点:
		1. 基于TCP/IP的高级协议
		2. 默认端口号:80
		3. 基于请求/响应模型的:一次请求对应一次响应
		4. 无状态的:每次请求之间相互独立,不能交互数据

	* 历史版本:
		* 1.0:每一次请求响应都会建立新的连接
		* 1.1:复用连接
*请求消息:客户端发送给服务器端的数据
* 请求消息数据格式
	1. 请求行
		请求方式 请求url 请求协议/版本
		GET /login.html	HTTP/1.1

		* 请求方式:
			* HTTP协议有7中请求方式,常用的有2种
				* GET:
					1. 请求参数在请求行中,在url后。
					2. 请求的url长度有限制的
					3. 不太安全
				* POST:
					1. 请求参数在请求体中
					2. 请求的url长度没有限制的
					3. 相对安全
	2. 请求头:客户端浏览器告诉服务器一些信息
		请求头名称: 请求头值
		* 常见的请求头:
			1. User-Agent:浏览器告诉服务器,我访问你使用的浏览器版本信息
				* 可以在服务器端获取该头的信息,解决浏览器的兼容性问题

			2. Referer:http://localhost/login.html
				* 告诉服务器,我(当前请求)从哪里来?
					* 作用:
						1. 防盗链:防止盗取链接
						2. 统计工作:统计从各个页面跳转过来的次数
	3. 请求空行
		空行,就是用于分割POST请求的请求头,和请求体的。
	4. 请求体(正文):
		* 封装POST请求消息的请求参数的

	* 字符串格式:
		POST /login.html	HTTP/1.1
		Host: localhost
		User-Agent: Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:60.0) Gecko/20100101 Firefox/60.0
		Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
		Accept-Language: zh-CN,zh;q=0.8,zh-TW;q=0.7,zh-HK;q=0.5,en-US;q=0.3,en;q=0.2
		Accept-Encoding: gzip, deflate
		Referer: http://localhost/login.html
		Connection: keep-alive
		Upgrade-Insecure-Requests: 1
		
		username=zhangsan	

*响应消息:服务器端发送给客户端的数据
* 响应消息数据格式
    1.响应行
        1.组成:协议/版本 响应状态码 状态码描述
        2.响应状态码:服务器告诉客户端浏览器本次请求和响应的一个状态。
            1.状态吗都是3位数字
            2.分类:
                1.1xx:服务器接收客户端消息,但没有接收完成,等待一段时间后,发送1xx状态吗询问客户端是否还要继续发送请求
                2.2xx:成功。代表:200
                3.3xx:重定向。代表:302(重定向),304(访问缓存)
                4.4xx:客户端错误。(请求错误)
                    *代表:
                        *404:请求路径没有对应的资源)
                        *405:请求方式没有对应的doXxx()方法
                5.5xx:服务器端错误。代表:500(服务器内部出现异常)
                                    
    2.响应头
        1.格式:头名称:值
        2.常见的响应头
            1.Content-Type:服务器告诉客户端本次响应体数据的格式以及编码格式
            2.Content-disposition:服务器告诉客户端以什么格式打开响应体数据
                *值:
                    1.in-line:默认值,在当前页面内打开
                    2.attachment:以附件形式打开响应体。文件下载
    3.响应空行
    4.响应体:传输的数据
        
*响应消息的字符串格式
    HTTP/1.1 200 OK
    Set-Cookie: JSESSIONID=3D35076950F5736C6140430EA35DE870; Path=/0601; HttpOnly
    Content-Type: text/html;charset=UTF-8
    Content-Length: 102
    Date: Mon, 01 Jun 2020 06:26:29 GMT

    <html>
      <head>
        <title>$Title$</title>
      </head>
      <body>
      hello,response!!!
      </body>
    </html>
Request:
1. request对象和response对象的原理
	1. request和response对象是由服务器创建的。我们来使用它们
	2. request对象是来获取请求消息,response对象是来设置响应消息

2. request对象继承体系结构:	
	ServletRequest		--	接口
		|	继承
	HttpServletRequest	-- 接口
		|	实现
	org.apache.catalina.connector.RequestFacade 类(tomcat)

3. request功能:
	1. 获取请求消息数据
		1. 获取请求行数据
			* GET /day14/demo1?name=zhangsan HTTP/1.1
			* 方法:
				1. 获取请求方式 :GET
					* String getMethod()  
				2. (*)获取虚拟目录:/day14
					* String getContextPath()
				3. 获取Servlet路径: /demo1
					* String getServletPath()
				4. 获取get方式请求参数:name=zhangsan
					* String getQueryString()
				5. (*)获取请求URI:/day14/demo1
					* String getRequestURI():		/day14/demo1
					* StringBuffer getRequestURL()  :http://localhost/day14/demo1

					* URL:统一资源定位符 : http://localhost/day14/demo1	中华人民共和国
					* URI:统一资源标识符 : /day14/demo1					共和国(范围更广)
				
				6. 获取协议及版本:HTTP/1.1
					* String getProtocol()

				7. 获取客户机的IP地址:
					* String getRemoteAddr()
				
		2. 获取请求头数据
			* 方法:
				* (*)String getHeader(String name):通过请求头的名称获取请求头的值
				* Enumeration<String> getHeaderNames():获取所有的请求头名称
			
		3. 获取请求体数据:
			* 请求体:只有POST请求方式,才有请求体,在请求体中封装了POST请求的请求参数
			* 步骤:
				1. 获取流对象
					*  BufferedReader getReader():获取字符输入流,只能操作字符数据
					*  ServletInputStream getInputStream():获取字节输入流,可以操作所有类型数据

				2. 再从流对象中拿数据
			
			
	2. 其他功能:
		1. 获取请求参数通用方式:不论get还是post请求方式都可以使用下列方法来获取请求参数
			1. String getParameter(String name):根据参数名称获取参数值    username=zs&password=123
			2. String[] getParameterValues(String name):根据参数名称获取参数值的数组  hobby=xx&hobby=game
			3. Enumeration<String> getParameterNames():获取所有请求的参数名称
			4. Map<String,String[]> getParameterMap():获取所有参数的map集合
		    *中文乱码问题:
                *get方式:Tomcat 8 已经将get方式的乱码问题解决了
                *post方式:会乱码
                    *解决方法:在获取参数前,指定流的编码格式,设置request的编码request.setCharacterEncoding("utf-8")
    
        2.请求转发:一种在服务器内部的资源跳转方式
            1.步骤:
                1.通过request对象获取请求转发器对象:RequestDispatcher getRequestDispatcher(String path)
                2.使用RequestDispatcher对象来进行转发:forward(ServletRequest request,ServletResponse response)
            2.特点:
                1.浏览器地址栏路径不发生变化
                2.只能转发到当前服务器内部资源中
                3.转发是一次请求
        3.共享数据:
            *域对象:一个有作用范围的对象,可以在范围内共享数据
            *request域:代表一次请求的范围,一般用于请求转发的多个资源中共享数据
                *方法:
                    1.void setAttribute(String name,Object obj):存储数据
                    2.Object getAttribute(String name):获取键对应的值
                    3.void removeAttribute(String name):通过键移除键值对
        4.获取ServletContext:
            

    *BeanUtils工具类,简化数据封装
        *用于封装javabean的
        1.JavaBean:标准的Java类
            1.要求:
                1.类必须被public修饰
                2.必须提供空参的构造器
                3.成员变量必须使用private修饰
                4.提供公共的serter和getter方法
            2.功能:封装数据
        2.概念:
            1.成员变量:
            2.属性:setter和getter方法截取后的产物
                例如:getUsername()-->Username-->username
                    username就是属性
        3.方法:
            1.setProperty();
            2.getProperty();
                操作的是属性而不是成员变量(当属性和成员变量名字不相同的特殊情况时)
            3.populate(Object javabean,Map map);将map集合的键值对信息,封装到对应的javaBean对象中
Response:
*功能:设置响应消息
    1.设置响应行
        1.格式:HTTP/1.1 200 OK
        2.设置状态码:setStatus(int sc)
    2.设置响应头:setHeader(String name,String value)
        
    3.设置响应体:
        *使用步骤:
            1.获取输出流对象
                *字符输出流:PrintWriter getWriter()
                *字节输出流:ServletOutputStream getOutputStream()
            2.使用输出流将数据输出到客户端浏览器
案例:
1.完成重定向
    *重定向:资源跳转的一种方式
    *代码实现:
        *原理:
        //1.设置状态码为302
        response.setStatus(302);
        //2.设置响应头:location
        response.setHeader("location","/0610/servletDemo02");
        
        *response对象的重定向方法
         response.sendRedirect("/0601/servletDemo02");
    *重定向的特点:redirect
        1.地址栏发生变化
        2.重定向可以访问其他站点(服务器)的资源
        3.重定向是两次请求
    *转发的特点:forward
        1.转发地址栏路径不变
        2.转发只能访问当前服务器下的资源
        3.转发是一次请求,可以共享request域的数据
    *路径写法:
        1.路径分类:
            1.相对路径:通过相对路径不可以确定唯一资源
                *如:./index.html
                *不以/开头,以.开头的路径
                *规则:找到当前资源和目标资源的相对位置关系
                    *./:当前目录
                    *../:上一级目录
            2.绝对路径:通过绝对路径可以确定唯一资源
                *如http://localhost/0601/servletDemo02 可以简化为 ---> /0601/servletDemo02
                *以/开头的路径
                
                *规则:判断定义的路径是给谁用的:即是判断请求从哪儿发出
                    *给客户端浏览器使用:需要加虚拟目录(项目的访问路径)
                        *建议虚拟目录动态获取:request.getContextPath()
                        *常见的给客户端用的路径有:<a> <form> 重定向...
                    *给服务器用的:不需要加虚拟目录
                        *常见的给服务器端用的路径有:转发路径
2.服务器输出字符数据到浏览器
    *步骤:
        1.获取字符输入流
        2.输出数据
    *注意:
        *中文乱码问题:
            1.PrintWriter pw = response.getWriter();获取的流的默认编码时ISO-8859-1
            2.设置该流的默认编码
            3.告诉浏览器响应体使用的编码(在获取流之前设置)
                *response.setContentType("text/html;charset=utf-8");
                
3.服务器输出字节数据到浏览器
     *步骤:
        1.获取字节输出流
        2.输出数据
4。验证码
    1.本质:图片
    2.目的:防止恶意表单注册
ServletContext
1.概念:代表整个web应用,可以和程序的容器(服务器)通信(交互数据)
2.获取:
    1.通过request.getServletContext();获取
    2.通过HttpServlet获取:
        this.getServletContext();
3.功能:
    1.获取MIME类型:
        *MIME类型:在互联网通信过程中定义的一种文件数据类型
            *格式:大类型/小类型  text/html  image/jpeg 
        *获取:String getMimeType(String file);
    2.域对象:共享数据
        1.setAttribute(Sttring name,Object value)
        2.getAttribute(String name)
        3.removeAttribute(String name)
        
        *ServletContext对象范围:所有用户所有请求的数据
    3.获取文件的真实(服务器)路径(文件运行的路径)
        1.方法:String getRealPath(String path);
        //获取文件的真实路径(服务器路径)
            String realPath = context.getRealPath("/b.txt");//web目录下资源访问
            System.out.println(realPath);
            
    
            String c = context.getRealPath("/WEB-INF/c.txt");//WEB-INF下资源访问
            System.out.println(c);
    
            String a = context.getRealPath("/WEB-INF/classes/a.txt");//src资源访问
            System.out.println(a);
案例:文件下载
*需求:
    1. 页面显示超链接
	2. 点击超链接后弹出下载提示框
	3. 完成图片文件下载
*步骤:
    1.定义页面,编辑超链接hre属性,指向servlett,传递资源名称filename
    2.定义servlet
        1.获取文件名称
        2.使用字节输入流加载文件进内存
        3.指定response的响应头:content-disposition:attachment;filename=xxx
        4.将文件写出到response输出流
*问题:
    *中文文件名问题
        *解决思路:
            1.获取客户端使用的浏览器版本信息
            2.根据不同的版本信息,设置filename的编码方式不同