当Tomcat 通过 Socket API 读取 HTTP 请求(字符串), 并且按照 HTTP 协议的格式把字符串解析成HttpServletRequest 对象.
创建一个ShowRequest类
@WebServlet("/ShowRequest") public class ShowRequest extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { StringBuilder stringBuilder = new StringBuilder(); // 返回请求协议的名称和版本号 stringBuilder.append(req.getProtocol()); // 协议名称: HTTP 版本号: 1.1 stringBuilder.append("\n"); // stringBuilder.append("
"); // 放回请求的HTTP方法名称 stringBuilder.append(req.getMethod()); // GET stringBuilder.append("\n"); // stringBuilder.append("
"); // 从协议名称知道HTTP请求的第一行的查询字符串中, 返回该请求的URL的一部分 stringBuilder.append(req.getRequestURI()); // /3010/ShowRequest stringBuilder.append("\n"); // stringBuilder.append("
"); stringBuilder.append(req.getRequestURL()); // 返回整个地址 http://127.0.0.1:8080/0310/ShowRequest // stringBuilder.append("
"); stringBuilder.append("\n"); // 返回指示请求上下文的请求URI部分 stringBuilder.append(req.getContextPath()); // /3010 - stringBuilder.append("\n"); // stringBuilder.append("
"); // 返回包含路径后的请求URL中的查询字符串 stringBuilder.append(req.getQueryString()); // null 没有QueryString stringBuilder.append("\n"); // stringBuilder.append("
"); // 把请求的header 拼接 // 获取到所有的headerNames // 枚举 EnumerationheaderNames = req.getHeaderNames(); // has XXX 判断是否有下一个 while(headerNames.hasMoreElements()) { // 遍历获取到每一个header的name值 String name = headerNames.nextElement(); // next XXX获取下一个 // 查询header中name对应的值 String value = req.getHeader(name); stringBuilder.append(name + ", " + value); stringBuilder.append("\n"); } resp.getWriter().write(stringBuilder.toString()); } }
通过 smart Tomcat 部署程序并在浏览器中通过URL http://127.0.0.1:8080/0310/ShowRequest访问, 可以看到如下结果:
由于 QueryString在传入请求时, 并没有设置, 因此此处获取到的QueryString 为 null
另外需要注意的是, 此处我们设置的换行符 “\n” 并非一直都是正确生效的. 此时由于我们并没有设置准确的相应格式, 因此浏览器会在加载时"猜测"此时响应的文本是什么类型, 此处恰好认为是text/plain 可以正确读取换行.
若当设置响应格式为 text/html时, 此时 “\n” 换行符将无法正常解析
![在这里插入图片描述](https://img-blog.csdnimg.cn/6f60b9e0e02046858abc7ff58945531c.png
换成html对应的换行标签
时, 即可以正常识别换行标签
因此, 在通过浏览器访问时, 应当主动设置setCOntentType(“类型”, charset=“utf8”), 告诉浏览器按照什么类型, 什么编码方式解析
创建一个GetParameterServlet类, 假设浏览器请求如下
?studentID=10&studentName=张三
@WebServlet("/getParameter") public class GetParameterServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { String studentId = req.getParameter("studentId"); // getparameter会自动针对urlencode 的结果进行处理 String studentName = req.getParameter("studentName"); System.out.println(studentId); System.out.println(studentName); resp.getWriter().write(studentId + ", " + studentName); }
运行smart Tomcat 通过网页访问URL路径为http://127.0.0.1:8080/0310/getParameter?studentId=10&studentName=张三
在idea窗口中, “张三” 却没有乱码
说明此时浏览器对于 QueryString的解析出现了问题, 浏览器对于urlencode解析不正确, 需要手动设置响应格式告诉浏览器如何去解析
当设置响应格式后, 浏览器便能正确识别了
在刚刚创建的 GetParameter 类中, 重写 dopost 方法, 并构建请求
@Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 通过 body 获取, 发送一个post请求 // 预期请求的 body 里为 studentId = 10&studentName = 张三 // 请求设置请求字符集 告诉 servlet(Tomcat) 该如何解析 req.setCharacterEncoding("utf8"); String studentId = req.getParameter("studentId"); String studentName = req.getParameter("studentName"); System.out.println(studentId); System.out.println(studentName); // 设置浏览器响应格式 告诉浏览器该如何解析 resp.setContentType("text/html; charset=utf8"); resp.getWriter().write(studentId + "," + studentName); }
通过 postman 发送一个 post 请求观察结果
在idea上 返回了 结果
采用 fiddler 观察抓包结果
创建一个新类 JacksonServlet 重写 dopost 方法
在中央仓库中(中央仓库地址)引入 Jackson 依赖
将 maven 下的内容复制到 pom.xml 文件中
com.fasterxml.jackson.core jackson-databind 2.13.4.1
编写代码
class Student { // 这个类里的属性必须是public 或者带有 public 的getter / setter 否则json无法访问 public int studentId; public String studentName; // 所构造的对象必须是 body 里构造的相同(名字与类型都应相同) // 里面的构造方法不写即为无参版本构造器 且必须为无参版本 } @WebServlet("/json") public class JacksonServlet extends HolleServlet{ @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 从请求 body 里面获取一些参数 // 例如格式为 {"studentId": 10 , "studentName" : 张三 } // 一个重要类 ObjectMapper objectMapper = new ObjectMapper(); // 第一个参数可以使字符串 也可以是输入流 第二个参数是一个类对象即为要解析的结果的类对象 Student s = objectMapper.readValue(req.getInputStream(), Student.class); // 类对象可以解析到类里的属性 // 1.读取输入流, 获取到需要解析的字符串 // 2. 把字符串按照json格式解析, 得到一组键值对 // 3. 根据类对象创建一个实例 // 4. 遍历类对象中的属性的名字, 拿着名字去键值对中查询, 查到的value就赋值到对应的对象属性中 // 5. 返回这个构造完成的对象 System.out.println(s.studentId); System.out.println(s.studentName); // 两个重要方法 readValue方法, 把 json 格式数据转成 java 的对象 // writerValueAsString方法, 把 java 对象转为 json 格式的字符串 // 设置 响应时, 浏览器按照 json 格式解析 resp.setContentType("application/json; charset=utf8"); // 先把s对象转为json格式字符串在转为流对象 resp.getWriter().write(objectMapper.writeValueAsString(s)); } }
使用 postman 构建 body 为 json 的 post 请求
idea 中接受到的 响应
利用 fiddler 抓包如下