Servlet的体系结构
Servlet接口下面有一个子类GenericServlet,而GenericServlet抽象类下又有一个子类HttpServlet抽象类。
GenericServlet抽象类
- 将Servlet接口中其他的方法做了默认空实现,只将service()方法作为抽象,将来定义Servlet类时,可以继承GenericServlet,实现service()方法即可
HttpServlet抽象类(推荐)
- HttpServlet:对http协议的一种封装,简化操作
- 定义类继承HttpServlet
- 复写doGet/doPost方法
Servlet相关配置
- urlpartten:Servlet访问路径
一个Servlet可以定义多个访问路径 : @WebServlet({"/d4","/dd4","/ddd4"}) 我们可以通过其中的任意一个配置 去访问对应的资源
路径定义规则:
- /xxx:路径匹配
- /xxx/xxx:多层路径,目录结构
- *.do:扩展名匹配
package cn.itcast.web.servlet; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; /** * Servlet路径配置 */ //@WebServlet({"/d4","/dd4","/ddd4"}) //@WebServlet("/user/demo4") //@WebServlet("/user/*") //@WebServlet("/*") @WebServlet("*.do") public class ServletDemo4 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("demo4..."); System.out.println(req); } }
HTTP协议
概念:
Hyper Text Transfer Protocol 超文本传输协议:定义了,客户端和服务器端通信时,发送数据的格式
特点:
- 基于TCP/IP的高级协议
- 默认端口号:80
- 基于请求/响应模型的:一次请求对应一次响应
- 无状态的:每次请求之间相互独立,不能交互数据
历史版本:
- 1.0:每一次请求响应都会建立新的连接
- 1.1:复用连接
请求消息数据格式
请求消息数据格式分为请求行,请求头,请求空格,请求体四部分。
请求行
请求行又包括:请求方式 请求url 请求协议/版本 例如:GET /login.html HTTP/1.1
HTTP协议有7中请求方式,常用的有2种
GET:
- 请求参数在请求行中,也就是在url后面。
- 请求的url长度有限制的
- 不太安全
POST:
- 请求参数在请求体中
- 请求的url长度没有限制的
- 相对安全
请求头:
请求头中的数据都是以键值对的形式存在主要作用:客户端浏览器告诉服务器一些信息
常见的请求头:
- User-Agent:浏览器告诉服务器,我访问你使用的浏览器版本信息,可以在服务器端获取该头的信息,解决浏览器的兼容性问题
- Referer:告诉服务器,我(当前请求)从哪里来?作用:防盗链or 统计工作
注意:
- get请求只要请求行和请求头,不存在请求空格和请求体。
请求空行
- 空行,就是用于分割POST请求的请求头,和请求体的。
请求体(正文):
- 封装POST请求消息的请求参数的
Request对象概述:
- request和response对象是由服务器创建的。我们来使用它们,request对象是来获取请求消息,response对象是来设置响应消息
Reques对象的体系结构
- org.apache.catalina.connector.RequestFacade 类由tomcat编写,它用来实现HttpServletRequest 接口,而HttpServletRequest 接口又是ServletRequest 接口的子接口
request功能(重中之重)
request的常用方法如下:
获取请求消息数据
获取客户端的请求方式 是get还是post,还是别的什么类型
- String getMethod()
获取项目所作的虚拟目录
- String getContextPath()
获取Servlet路径:
- String getServletPath()
获取get方式所有请求参数:
- String getQueryString()
获取请求URI:
- String getRequestURI(): 获取URI。URI:统一资源标识符
- StringBuffer getRequestURL() :获取URL。 URL:统一资源定位符
获取协议及版本:
- String getProtocol()
获取客户机的IP地址:
- String getRemoteAddr()
举例
我们先准备好测试代码
package DemoRequest; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; @WebServlet("/TestServlet") public class TestServlet extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //1. 获取请求方式 :GET String method = request.getMethod(); System.out.println(method); //2.(*)获取虚拟目录:/day14 String contextPath = request.getContextPath(); System.out.println(contextPath); //3. 获取Servlet路径: /demo1 String servletPath = request.getServletPath(); System.out.println(servletPath); //4. 获取get方式请求参数:name=zhangsan String queryString = request.getQueryString(); System.out.println(queryString); //5.(*)获取请求URI:/day14/demo1 String requestURI = request.getRequestURI(); StringBuffer requestURL = request.getRequestURL(); System.out.println(requestURI); System.out.println(requestURL); //6. 获取协议及版本:HTTP/1.1 String protocol = request.getProtocol(); System.out.println(protocol); //7. 获取客户机的IP地址: String remoteAddr = request.getRemoteAddr(); System.out.println(remoteAddr); } }
我们通过浏览器去访问http://localhost:8080/Test/TestServlet?name=zhangsan&age=13得到的结果是
获取请求头数据
- String getHeader(String name):通过请求头的名称获取请求头的值
- Enumeration<String> getHeaderNames():获取所有的请求头名称
package DemoRequest; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.util.Enumeration; @WebServlet("/Demo2") public class RequestDemo2 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //演示获取请求头数据 //1.获取所有请求头名称 Enumeration<String> headerNames = request.getHeaderNames(); //2.遍历 while (headerNames.hasMoreElements()) { String name = headerNames.nextElement(); //根据名称获取请求头的值 String value = request.getHeader(name); System.out.println(name + "---" + value); } } }
访问后的结果
获取请求体数据:
请求体:只有POST请求方式,才有请求体,在请求体中封装了POST请求的请求参数
步骤:
1. 获取流对象
- BufferedReader getReader():获取字符输入流,只能操作字符数据
- ServletInputStream getInputStream():获取字节输入流,可以操作所有类型数据
2. 再从流对象中拿数据
import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.BufferedReader; import java.io.IOException; @WebServlet("/requestDemo5") public class RequestDemo5 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //获取请求消息体--请求参数 //1.获取字符流 BufferedReader br = request.getReader(); //2.读取数据 String line = null; while((line = br.readLine()) != null){ System.out.println(line); } } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { } }
其他功能:
获取请求参数通用方式:不论get还是post请求方式都可以使用下列方法来获取请求参数
- String getParameter(String name):根据参数名称获取参数值
- String[] getParameterValues(String name):根据参数名称获取参数值的数组 hobby=xx&hobby=game
- Enumeration<String> getParameterNames():获取所有请求的参数名称
- Map<String,String[]> getParameterMap():获取所有参数的map集合
中文乱码问题:
- get方式:tomcat 8 已经将get方式乱码问题解决了
- post方式:会乱码。解决:在获取参数前,设置request的编码request.setCharacterEncoding("utf-8");
import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.BufferedReader; import java.io.IOException; import java.util.Enumeration; import java.util.Map; import java.util.Set; @WebServlet("/requestDemo6") public class RequestDemo6 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //post 获取请求参数 //根据参数名称获取参数值 String username = request.getParameter("username"); /* System.out.println("post"); System.out.println(username);*/ //根据参数名称获取参数值的数组 String[] hobbies = request.getParameterValues("hobby"); /*for (String hobby : hobbies) { System.out.println(hobby); }*/ //获取所有请求的参数名称 Enumeration<String> parameterNames = request.getParameterNames(); // 获取所有参数的map集合 Map<String, String[]> parameterMap = request.getParameterMap(); //遍历 Set<String> keyset = parameterMap.keySet(); for (String name : keyset) { //获取键获取值 String[] values = parameterMap.get(name); System.out.println(name); for (String value : values) { System.out.println(value); } System.out.println("-----------------"); } } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //get 获取请求参数 this.doPost(request,response); } }
请求转发:一种在服务器内部的资源跳转方式
步骤:
- 通过request对象获取请求转发器对象,使用的方法:RequestDispatcher getRequestDispatcher(String path)
- 使用RequestDispatcher对象来进行转发:使用方法:forward(ServletRequest request, ServletResponse response)
特点:
- 浏览器地址栏路径不发生变化
- 只能转发到当前服务器内部资源中。
- 转发是一次请求
import javax.servlet.RequestDispatcher; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; @WebServlet("/requestDemo8") public class RequestDemo8 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("demo8888被访问了。。。"); //转发到demo9资源 request.getRequestDispatcher("/requestDemo9").forward(request,response); //request.getRequestDispatcher("http://www.itcast.cn").forward(request,response); 错误只能转发到当前服务器内部资源中。 } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request,response); } }
共享数据:
- 域对象:一个有作用范围的对象,可以在范围内共享数据
- request域:代表一次请求的范围,一般用于请求转发的多个资源中共享数据
方法:
- void setAttribute(String name,Object obj):存储数据
//存储数据到request域中 request.setAttribute("msg","hello");
- Object getAttitute(String name):通过键获取值
- void removeAttribute(String name):通过键移除键值对
获取ServletContext:
- ServletContext getServletContext()