ServletRequest 基本概念
JavaWeb中的 "Request"对象
实际为 HttpServletRequest 或者 ServletRequest, 两者都为接口
服务器接收请求后,将请求数据进行对象封装
功能大致分类
request的功能可以分为以下几种:
- 封装了请求头数据;
- 封装了请求正文数据,如果是GET请求,那么就没有正文;
- request是一个域对象,可以把它当成Map来添加获取数据;
- request提供了请求转发和请求包含功能。
request是四大域对象之一
其他是ServletContext Session PageContext
Request的域方法
用来存储一个对象,也可以称之为存储一个域属性
| void setAttribute(String name, Object value);
|
例如:servletContext.setAttribute(“xxx”, “XXX”),在request中保存了一个域属性,域属性名称为xxx,域属性的值为XXX。
请注意,如果多次调用该方法,并且使用相同的name,那么会覆盖上一次的值,这一特性与Map相同;
用来获取request中的数据
| Object getAttribute(String name);
|
当前在获取之前需要先去存储才行,
例如:
String value = (String)request.getAttribute(“xxx”);,获取名为xxx的域属性;
用来移除request中的域属性,如果参数name指定的域属性不存在,那么本方法什么都不做;
| void removeAttribute(String name);
|
获取所有域属性的名称;
| Enumeration getAttributeNames();
|
获取请求头数据
获取指定名称的请求头;
| String getHeader(String name);
|
获取所有请求头名称;
| Enumeration getHeaderNames();
|
获取值为int类型的请求头。
| int getIntHeader(String name);
|
获取请求相关的其它方法
还提供了与请求相关的其他方法,有些方法是为了我们更加便捷的方法请求头数据而设计,有些是与请求URL相关的方法
获取请求体的字节数,GET请求没有请求体,没有请求体返回-1;
获取请求类型,如果请求是GET,那么这个方法返回null;
如果是POST请求,那么默认为application/x-www-form-urlencoded,表示请求体内容使用了URL编码;
返回请求方法,例如:GET
返回当前客户端浏览器的Locale。java.util.Locale表示国家和言语,这个东西在国际化中很有用;
获取请求体编码,如果没有setCharacterEncoding(),那么返回null,表示使用ISO-8859-1编码;
| String getCharacterEncoding();
|
设置请求编码,只对请求体有效!注意,对于GET而言,没有请求体!!!所以此方法只能对POST请求中的参数有效!
| void setCharacterEncoding(String code);
|
返回主机名,例如:localhost 或者 127.0.0.1
返回请求URI路径,例如:/servlet/ServletA
返回请求URL路径,例如:http://localhost:8080/servlet/ServletA,
即返回除了参数以外的路径信息;
| StringBuffer getRequestURL();
|
返回请求URL中的参数,例如:name=zhangsan&age=28
返回上下文路径,例如:/servlet
返回Servlet路径,例如:/ServletA
返回当前客户端的IP地址;
返回当前客户端的主机名,但这个方法的实现还是获取IP地址;
上面的路径示例 假定请求地址为:
http://127.0.0.1:8080/servlet/ServletA?name=zhangsan&age=28
获取请求参数
最为常见的客户端传递参数方式有两种GET 和 POST:
浏览器地址栏直接输入:一定是GET请求;
超链接:一定是GET请求;
表单:可以是GET,也可以是POST,这取决与<form>的method属性值;
|
请求参数会在浏览器的地址栏中显示,所以不安全;
请求参数长度限制长度在1K之内;
GET请求没有请求体,无法通过request.setCharacterEncoding()来设置参数的编码;
|
|
请求参数不会显示浏览器的地址栏,相对安全;
请求参数长度没有限制;
|
获取请求参数的具体方法
根据参数名称获取参数
| public String getParameter(String name); |
页面一个超链接一个表单
<body>
<a href="/servlet/ServletA?name=zhangsan&age=28">超链接</a>
<hr/>
<form action="/servlet/ServletA" method="post">
参数1:<input type="text" name="name"/><br/>
参数2:<input type="text" name="age"/><br/>
<input type="submit" value="提交"/>
</form>
</body>
Servlet中java代码(doGet doPost 都一样可以)
System.out.println("request.getParameter(\"name\"): " + request.getParameter("name"));
System.out.println("request.getParameter(\"age\"): " + request.getParameter("age"));
打印结果:
当多个参数名称相同时,可以使用方法来获取
| String[] getParameterValues(String name); |
http://127.0.0.1:8080/servlet/ServletA?name=zhangsan&name=lisi
System.out.println("request.getParameter(\"name\"): " + request.getParameter("name"));
String[] names = request.getParameterValues("name");
System.out.println(Arrays.toString(names));
打印结果:
可以看得出来request.getParameter("name") 打印的是第一个
获取所有参数的名称
| public Enumeration getParameterNames(); |
http://127.0.0.1:8080/servlet/ServletA?name=zhangsan&age=28&sex=male
Enumeration pNames = request.getParameterNames();
while(pNames.hasMoreElements()) {
System.out.println(pNames.nextElement());
}
打印信息
获取所有参数封装到Map中,其中key为参数名,value为参数值
因为一个参数名称可能有多个值,所以参数值是String[],而不是String
| public Map getParameterMap(); |
http://127.0.0.1:8080/servlet/ServletA?name=zhangsan&age=28&sex=male&name=lisi&age=18
//转换为Map
Map<String,String[]> paramMap = request.getParameterMap();
//获取键,根据键获取值数组
for(String name : paramMap.keySet()) {
String[] values = paramMap.get(name);
System.out.println(name + ": " + Arrays.toString(values));
}
打印结果:
请求转发和请求包含
RequestDispatcher forward include
无论是请求转发还是请求包含,都表示由多个Servlet共同来处理一个请求。
请求转发与请求包含比较
1.如果在AServlet中请求转发到BServlet,那么在AServlet中就不允许再输出响应体,
即不能再使用response.getWriter()和response.getOutputStream()向客户端输出,这一工作应该由BServlet来完成;
如果是使用请求包含,那么没有这个限制;
2.请求转发虽然不能输出响应体,但还是可以设置响应头的
例如:response.setContentType(”text/html;charset=utf-8”);
3.请求包含大多是应用在JSP页面中,完成多页面的合并;
4.请求转发大多是应用在Servlet中,转发目标大多是JSP页面;
ServletA中代码如下:
response.getWriter().println("ServletA");
//获取“调度器”,其中参数为BServlet绑定的URL,即BServlet的<url-pattern>值。
RequestDispatcher rd = request.getRequestDispatcher("/ServletB");
rd.forward(request, response);
B中仅仅响应信息
请求http://127.0.0.1:8080/servlet/ServletA
页面数据为:
可以看得到,虽然A中有response.getWriter().println("ServletA"); 但是没有任何的结果,请求直接被转发了
请求 http://127.0.0.1:8080/servlet/ServletA
ServletA中代码如下:
response.getWriter().println("ServletA");
//获取“调度器”,其中参数为BServlet绑定的URL,即BServlet的<url-pattern>值。
RequestDispatcher rd = request.getRequestDispatcher("/ServletB");
rd.include(request, response);
页面数据为:
此时A和B Servlet的响应都可以正常输出,也就是响应进行了合并
请求转发与重定向比较
请求转发是一个请求,而重定向是两个请求;
请求转发后浏览器地址栏不会有变化,而重定向会有变化,因为重定向是两个请求;
请求转发的目标只能是本应用中的资源,重定向的目标可以是其他应用;
请求转发对ServletA和ServletB的请求方法是相同的,即要么都是GET,要么都是POST,因为请求转发是一个请求;
重定向的第二个请求一定是GET;
- HTTP原理 请求方法
HTTP的工作过程 一次HTTP操作称为一个事务,其工作过程分为四步: 1.客户机与服务器建立连接:客户单击某个超级链接,HTTP的工作开始,接下来进行TCP连接的三次握手过程. 2.建立连接后,客户 ...
- HTTP协议----请求方法和状态码
现在广泛使用的是HTTP/1.1版本,发布于1997年. 理解HTTP协议,首先从请求开始,比如: POST /form/entry HTTP/1.1 格式为: 请求方法 URI 协议版本 请求方法: ...
- [转帖]HTTP请求方法:GET、HEAD、POST、PUT、DELETE、CONNECT、OPTIONS、TRACE 说明
HTTP请求方法:GET.HEAD.POST.PUT.DELETE.CONNECT.OPTIONS.TRACE 说明 平时的Rest开发,用到的都是GET,POST,PUT,DELETE类型的请求. ...
- HTTP请求方法:GET、HEAD、POST、PUT、DELETE、CONNECT、OPTIONS、TRACE 说明
平时的Rest开发,用到的都是GET,POST,PUT,DELETE类型的请求. 但Rest支持的请求类型不止前面4种,还有其他几种. 下面部分转自: https://www.html.cn/arch ...
- HTTP请求方法及常见状态码
GET: 请求指定的页面信息,并返回实体主体. HEAD: 只请求页面的首部. POST: 请求服务器接受所指定的文档作为对所标识的URI的新的从属实体. PUT: 从客户端向服务器传送的数据取代指定 ...
- springboot 获取控制器参数的几种方式
这里介绍springboot 获取控制器参数有四种方式 1.无注解下获取参数 2.使用@RequestParam获取参数 3.传递数组 4.通过URL传递参数 无注解下获取参数无注解下获取参数,需要控 ...
- javascript获取URL参数和参数值
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- Spring 请求方法的调用原理(Controller)和请求参数的获取的原理
1.请求映射原理 所有的请求都会经过DispatcherServlet这个类,先了解它的继承树 本质还是httpServlet 原理图 测试 request请求携带的参数 从requestMapp ...
- SpringMVC04 很杂很重要(注解,乱码处理,通配符,域属性调用,校正参数名称,访问路径,请求、响应携带参数,请求方法)
1.导入架包 <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3 ...
随机推荐
- GitLab在centos7上安装和使用
git的优点 git是分布式的,svn不是 git分布式本地就可以用,可以随便保存各种历史痕迹,不用担心污染服务器,连不上服务器也能提交代码.查看log. GIT分支和SVN的分支不同 分支在SVN中 ...
- 基于.net的Socket异步编程总结
最近在为公司的分布式服务框架做支持异步调用的开发,这种新特性的上线需要进行各种严格的测试.在并发性能测试时,性能一直非常差,而且非常的不稳定.经过不断的分析调优,发现Socket通信和多线程异步回调存 ...
- JavaScript模板引擎Handlebars
Handlebars模板库简单介绍 Handlebars是JavaScript一个语义模板库,通过对view(模板)和data(ajax请求的数据,一般是json)的分离来快速构建Web模板.它采用& ...
- [zt]C++二维数组讲解、二维数组的声明和初始化
定义: int *pia = new int[10]; // array of 10 uninitialized ints 此 new 表达式分配了一个含有 10 个 int 型元素的数组,并返回指向 ...
- vue常用属性
关键词:filters | 自定义过滤器 (首字母大写) <p>{{ msg | capitalize }}</p> filters: { capitalize: funct ...
- linux学习历程-不熟悉的linux命令
一:man(执行查看帮助命令) 二:常用的系统工作命令 1:echo echo命令用于显示在终端输出字符串或变量提取后的值,格式“echo [字符串]|[$变量]” 2:date 用于显示系统的时间和 ...
- Python基础之面向对象2(封装)
一.封装定义: 二.作用 三.私有成员: 1.基本概念及作用 2.__slots__手段私有成员: 3.@property属性手段私有成员: 四.基础示例代码 1.用方法封装变量 "&quo ...
- angular.js学习笔记(一)
1.angular单项数据绑定 2.不要使用控制器的时候: 任何形式的DOM操作:控制器只应该包含业务逻辑.DOM操作则属于应用程序的表现层逻辑操作,向来以测试难度之高闻名于业界.把任何表现层的逻辑放 ...
- 1.2 Why need pluggable?
When Android programmers write new features, bugs, or even crashes will exits in their App. Once a t ...
- 对某菠菜网站的一次渗透测试 heatlevel
前言 无意间发现一个thinkphp的菠菜站,最近tp不是刚好有个漏洞吗?然后就顺手测试了一下,但过程并不太顺利,不过最后还是拿下了,所以特发此文分享下思路. 0x00 一键getshell? 简单看 ...