1.HttpServletRequest概述
我们在创建Servlet时会覆盖service()方法,或doGet()/doPost(),这些方法都有两个参数,一个为代表请求的request和代表响应response。
service方法中的request的类型是ServletRequest,而doGet/doPost方法的request的类型是HttpServletRequest,HttpServletRequest是ServletRequest的子接口,功能和方法更加强大.
3.通过抓包工具抓取Http请求
因为request代表请求,所以我们可以通过该对象分别获得Http请求的请求行,请求头和请求体。
4.通过request获得请求行
获得客户端的请求方式:String getMethod()
获得请求的资源:
String getRequestURI() ----通常代指所有的地址,报错本地磁盘路径和网络地址等
StringBuffer getRequestURL() ----通常代指网络地址
String getContextPath() ---web应用的名称
String getQueryString() ---- get提交url地址后的参数字符串 username=zhangsan&password=123
注意:request获得客户机(客户端)的一些信息
request.getRemoteAddr()
— 获得访问的客户端IP地址
//各个方法对应输出示例:
Method=GET
URI=/Test_Servlet/login
URL=http://localhost:8080/Test_Servlet/login
Path=/Test_Servlet
get的参数=username=&password=
5.通过request获得请求头
long getDateHeader(String name)
String getHeader(String name)
Enumeration getHeaderNames()
Enumeration getHeaders(String name)
int getIntHeader(String name)
例如通过getHeader()和getHeaderNames()方法来获得请求头信息
Enumeration<String> headerNames = request.getHeaderNames(); //Enumeration是枚举类型
while (headerNames.hasMoreElements()) {
String headerName = (String) headerNames.nextElement();
String headerValue = request.getHeader(headerName) ;
System.out.println(headerName+":"+headerValue);
}
输出结果:
host:localhost:8080
connection:keep-alive
upgrade-insecure-requests:1
user-agent:Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/65.0.3325.181 Safari/537.36
accept:text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,/;q=0.8
referer:http://localhost:8080/Test_Servlet/
accept-encoding:gzip, deflate, br
accept-language:zh-CN,zh;q=0.9
cookie:JSESSIONID=2DB82255BD2B4EACCBC7C72B6CE88D33
其中,referer头的作用:执行该此访问的的来源 , 用于做防盗链
6.通过request获得请求体
请求体中的内容是通过post提交的请求参数,格式是:
username=zhangsan&password=123&hobby=football&hobby=basketball
以上面参数为例,通过一下方法获得请求参数:
String getParameter(String name)
String[] getParameterValues(String name)
Enumeration getParameterNames()
Map<String,String[]> getParameterMap()
request.getParameter(username) --------zhangsan
request.getParameterMap()---------
key ---------------------- value
username [zhangsan]
password [123]
hobby [football,basketball]
注意:get请求方式的请求参数 上述的方法一样可以获得
解决post提交方式的乱码:request.setCharacterEncoding("UTF-8");
解决get提交的方式的乱码:parameter = new String(parameter.getbytes("iso8859-1"),"utf-8");
7.request的其他功能
1)request是一个域对象
request对象也是一个存储数据的区域对象,所以也具有如下方法:
setAttribute(String name, Object o)
getAttribute(String name)
removeAttribute(String name)
注意:request域的作用范围:一次请求中(因此当重定向之后request域对象里面的数据会消失)
2)request完成请求转发
获得请求转发器—-path是转发的地址
RequestDispatcher getRequestDispatcher(String path)
通过转发器对象转发
requestDispathcer.forward(ServletRequest request, ServletResponse response)
实例:
/**
* Servlet名:TestForwardServlet1
* @WebServlet("/forward")
* 监听/forward请求,并且将该请求转发到/forward2请求,转发前先对request添加数据,此数据在该次请求中会一直存在,因此TestForwardServlet2中会收到
/
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.getWriter().append("Served at: ").append(request.getContextPath());
//对域对象request添加数据
request.setAttribute("name", "zhangchi");
RequestDispatcher requestDispatcher = request.getRequestDispatcher("/forward2");
requestDispatcher.forward(request, response);
}
/**
* Servlet名:TestForwardServlet2
* @WebServlet("/forward2")
* 收到TestForwardServlet1转发的请求,并且同时接收到request发送的数据,使用getAttribute()方法将数据提出来
/
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//取出域对象的数据值
response.getWriter().append("Served at: ").append(request.getContextPath()).append("------name="+request.getAttribute("name"));
}
3)请求转发和重定向
注意:转发与重定向的区别?
1)重定向两次请求,转发一次请求
2)重定向地址栏的地址变化,转发地址不变
3)重新定向可以访问外部网站 转发只能访问内部资源
4)转发的性能要优于重定向
4)ServletContext域与Request域的生命周期比较?
ServletContext:
创建:服务器启动
销毁:服务器关闭
域的作用范围:整个web应用
request:
创建:访问时创建request
销毁:响应结束request销毁
域的作用范围:一次请求中
5)客户端地址与服务器端地址的写法?
客户端地址:
是客户端去访问服务器的地址,服务器外部的地址,特点:写上web应用名称
直接输入地址:
重定向
服务器端地址:
服务器内部资源的跳转的地址,特点:不需要写web应用的名称
请求转发
8.案例——注册功能
@WebServlet("/register")
public class RegisterServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.getWriter().append("Served at: ").append(request.getContextPath());
//设置request的编码---只适合post方式
request.setCharacterEncoding("UTF-8");
// 1.获取用户输入的用户名和密码
// String username = request.getParameter("username");
// String password = request.getParameter("password");
// String email = request.getParameter("email") ;
// String birthday = request.getParameter("birthday") ;
//get方式的解决乱码
//先用iso8859-1编码 在使用utf-8解码
// username = new String(username.getBytes("iso8859-1"),"UTF-8");
//使用BeanUtils进行自动映射封装
//BeanUtils工作原理:将map中的数据 根据key与实体的属性的对应关系封装
//只要key的名字与实体的属性 的名字一样 就自动封装到实体中
Map parameterMap = request.getParameterMap();
User user = new User() ;
try {
BeanUtils.populate(user, parameterMap);
} catch (IllegalAccessException | InvocationTargetException e1) {
e1.printStackTrace();
}
//3、将参数传递给一个业务操作方法
try {
regist(user);
} catch (Exception e) {
e.printStackTrace();
}
//4、认为注册成功跳转到登录页面
response.sendRedirect(request.getContextPath()+"/index.jsp");
}
//业务方法——注册
public void regist(User user){
QueryRunner queryRunner = new QueryRunner(DataSourceUtils.getDataSource());
String sql = "insert into user values(null,?,?,?,?)";
try {
queryRunner.update(sql,user.getUsername(), user.getPassword(),user.getEmail(),user.getBirthday());
} catch (SQLException e) {
e.printStackTrace();
}
}
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}
}
其中,通过导入common-beanutils和common-loggin两个jar包,使用BeanUtils工具类实现自动将请求体中的内容和User中的属性对应,前提是请求的key和属性名相同。这样可以节省获取请求中的参数的代码。