目录
一:域对象
1、应用域对象
2、请求域对象
二:request对象其它常用的方法
(1)应用域对象是什么?
ServletContext (Servlet上下文对象。)
什么情况下会考虑向ServletContext这个应用域当中绑定数据呢?
第一:所有用户共享的数据。
第二:这个共享的数据量很小。
第三:这个共享的数据很少的修改操作。
在以上三个条件都满足的情况下,使用应用域对象,可以大大提高我们程序执行效率。
实际上向应用域当中绑定数据,就相当于把数据放到了缓存(Cache)当中,然后用户访问的时候直接从缓存中取,减少IO的操作,大大提升系统的性能,所以缓存技术是提高系统性能的重要手段。
(2)常见的缓存技术?
字符串常量池
整数型常量池 [-128~127],在这个范围当中的Integer对象不再创建新对象,直接从这个整数型常量池中获取,大大提升系统性能。
数据库连接池(提前创建好N个连接对象,将连接对象放到集合当中,使用连接对象的时候,直接从缓存中拿。省去了连接对象的创建过程,效率提升)
线程池(Tomcat服务器就是支持多线程的。所谓的线程池就是提前先创建好N个线程对象,将线程对象存储到集合中,然后用户请求过来之后,直接从线程池中获取线程对象,直接拿来用。提升系统性能)
后期还会学习更多的缓存技术,例如:redis、mongoDB.....
(3) ServletContext当中有三个操作域的方法:
void setAttribute(String name, Object obj); // 向域当中绑定数据
Object getAttribute(String name); // 从域当中根据name获取数据
void removeAttribute(String name); // 将域当中绑定的数据移除
以上的操作很像Map集合的操作Map
map.put("name", obj); // 向map集合中放key和value
Object obj = map.get("name"); // 通过map集合的key获取value
map.remove("name"); // 通过Map集合的key删除key和value这个键值对
request对象实际上又称为“请求域”对象!
(1)“请求域”对象
“请求域”对象要比“应用域”对象作用范围小很多,生命周期短很多。请求域只在一次请求内有效。
一个请求对象request对应一个请求域对象;一次请求结束之后,这个请求域就销毁了。
请求域对象也有这三个方法:
void setAttribute(String name, Object obj); // 向域当中绑定数据。 Object getAttribute(String name); // 从域当中根据name获取数据。 void removeAttribute(String name); // 将域当中绑定的数据移除
请求域和应用域的选用原则?
尽量使用小的域对象,因为小的域对象占用的资源较少。
(2) 方法测试
①定义Aservlet继承HttpServlet,发送Get请求
package com.bjpowernode.javaweb.servlet;import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Date;public class Aservlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {response.setContentType("text/html;charset=utf-8");PrintWriter out = response.getWriter();// 获取系统当前时间Date nowDate = new Date();// 向request域中绑定数据request.setAttribute("sysTime",nowDate);// 从request域当中取出绑定的数据Object obj = request.getAttribute("sysTime");// 输出打印到浏览器上// request域当中获取的系统当前时间 = Tue Nov 08 17:26:19 CST 2022out.print("request域当中获取的系统当前时间 = " + obj);}
}
web.xml文件进行配置
aservlet com.bjpowernode.javaweb.servlet.Aservlet aservlet /a
启动Tomcat服务器,进行访问:http://localhost:8080/servlet09/a,可以访问到;说明成功向请求域中绑定数据和请求数据
② 定义Bservlet继承HttpServlet,也发送Get请求;这次让Aservlet只绑定数据,用Bservlet取
package com.bjpowernode.javaweb.servlet;import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;public class Bservlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {// 输出到浏览器response.setContentType("text/html");PrintWriter out = response.getWriter();// 从request域当中取出绑定的数据。Object obj = request.getAttribute("sysTime");out.print("request域当中获取的系统当前时间 = " + obj);}
}
web.xml中添加配置文件
package com.bjpowernode.javaweb.servlet;import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;public class Bservlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {// 输出到浏览器response.setContentType("text/html");PrintWriter out = response.getWriter();// 从request域当中取出绑定的数据。Object obj = request.getAttribute("sysTime");out.print("request域当中获取的系统当前时间 = " + obj);}
}
先访问http://localhost:8080/servlet09/a 绑定数据,然后在访问http://localhost:8080/servlet09/b 取数据;发现Bservlet取出来的是null
③结论:说明Aservlet和Bservlet不是同一个servlet,一个对象对应一个servlet。但是可以使用Servlet当中请求转发机制,通过AServlet跳转到BServlet,让AServlet和BServlet放到一次请求当中。
(3)转发跳转机制
①我们首先可以这样考虑:在AServlet当中new一个BServlet对象,然后调用BServlet对象的doGet方法,把request对象传过去。这样虽然可以实现功能,但是Servlet对象不能自己由程序员来new,自己new的Servlet对象生命周期不受Tomcat服务器的管理。
BServlet bServlet = new BServlet();
bServlet.doGet(request, response);
②使用转发跳转机制
// 第一步:获取请求转发器对象
// 相当于把"/b"这个路径包装到请求转发器当中,实际上是把下一个跳转的资源的路径告知给Tomcat服务器
RequestDispatcher requestDispatcher = request.getRequestDispatcher("/b");
// 第二步:调用请求转发器RequestDispatcher的forward方法。进行转发。
// 转发的时候:这两个参数很重要。request和response都是要传递给下一个资源的。
requestDispatcher.forward(request,response);
// 代码合并
request.getRequestDispatcher("/b").forward(request,response);
③两个Servlet怎么共享数据?
(1)将数据放到ServletContext应用域当中,当然是可以的,但是应用域范围太大,占用资源太多,不建议使用。
(2)可以将数据放到request域当中,然后AServlet转发到BServlet,保证AServlet和BServlet在同一次请求当中,这样就可以做到两个Servlet,或者多个Servlet共享同一份数据
④转发的下一个资源必须是一个Servlet吗?
(1)不一定,只要是Tomcat服务器当中的合法资源,都是可以转发的。例如:html....
(2)注意:转发的时候,路径的写法要注意,转发的路径以“/”开始,不加项目名。
// 转发到一个Servlet,也可以转发到一个HTML,只要是WEB容器当中的合法资源即可。
request.getRequestDispatcher("/test.html").forward(request, response);
⑤关于request对象中两个非常容易混淆的方法:
第一个方法:获取的是用户在浏览器上提交的数据。
第二个方法:获取的是请求域当中绑定的数据。
// uri?username=zhangsan&userpwd=123&sex=1
String username = request.getParameter("username");// 之前一定是执行过:request.setAttribute("name", new Object())
Object obj = request.getAttribute("name");
(1)获取客户端的IP地址
目前Tomcat服务器和浏览器上都是在本机上,所以IP地址就是本机:127.0.0.1
// 获取客户端的IP地址
String remoteAddr = request.getRemoteAddr();
// 访问时用localhost:0:0:0:0:0:0:0:1
// 访问时用127.0.0.1: 127.0.0.1
// 0:0:0:0:0:0:0:1是ipv6的表现形式,对应ipv4来说相当于127.0.0.1,也就是本机
System.out.println(remoteAddr);
(2)处理POST请求的乱码
①这个方法是设置请求体的字符集。
②对于get请求在请求行上提交数据,对于post请求在请求体中提交数据。
③显然这个方法是处理POST请求的乱码问题,这种方式并不能解决get请求的乱码问题。
④注意:Tomcat10之后,request请求体当中的字符集默认就是UTF-8,不需要设置字符集,不会出现乱码问题。 Tomcat9前(包括9在内),如果前端请求体提交的是中文,后端获取之后就可能出现乱码,这就需要下面这段代码!
// 设置请求体的字符集
request.setCharacterEncoding("UTF-8");
补充: 在Tomcat9之前(包括9),响应中文也是有乱码的,怎么解决这个响应的乱码?
在Tomcat10之后,包括10在内,响应中文的时候就不在出现乱码问题了;以下代码就不需要设置UTF-8了。
response.setContentType("text/html;charset=UTF-8");
(3)处理GET请求的乱码
①get请求发送的时候,数据是在请求行上提交的,不是在请求体当中提交的。
②GET请求默认是不会发生乱码的;因为在CATALINA_HOME/conf/server.xml配置文件中有样一段代码;这段代码是默认的,就算不写也会有
③注意:从Tomcat8之后,URIEncoding的默认值就是UTF-8(不写也是默认),所以GET请求也没有乱码问题了
注:上面都是HttpServle从ServletRequest接口继承过来的方法;下面才是子类HttpServle独有的方法
(4)获取应用的根路径
这个方法使用比较多,可以动态获取应用的根路径!
// 获取应用的根路径
String contextPath = request.getContextPath();
System.out.println(contextPath); // /servlet09
(5)获取请求的方式
是GET请求还是POST请求?
// 获取请求的方式
String method = request.getMethod();
System.out.println(method); // GET
(5)获取请求的URL和URI
带项目名!
// 获取请求的URI
String requestURI = request.getRequestURI();
System.out.println(requestURI); // /servlet09/b
// 获取请求的URL
StringBuffer requestURL = request.getRequestURL();
System.out.println(requestURL); // http://localhost:8080/servlet09/b
(6)获取servlet path
不带项目名!
// 获取servlet path
String servletPath = request.getServletPath();
System.out.println(servletPath); // /b