当前位置: 首页 > 编程日记 > 正文

Request.getInputStrema只能读取一次的分析过程

1. 我们先来看一下继承关系HttpServletRequest 接口继承ServletRequest接口

public abstract interface  ServletRequest
{

public abstract ServletInputStream getInputStream()  throws IOException;

从上面可知request.getInputStream()返回的是ServletInputSteam。查看ServletInputStream源码可知,ServletInputStream extends InputSteam

2.为什么说request.getInputStream()只能读取一次呢,再读一次就没有内容了,这个问题要复习下java的IO知识了。

,在java中读取一个文件或者字符串的内容的代码大家都会写,下边是使用ByteArrayInputStream和ByteArrayOutputStream进行演示:

@Testpublic void testByteArrayInputStream() throws Exception {String str = "AAAAACCCCcCBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB";//ByteArrayInputStream是把一个byte数组转换成一个字节流,读取的内容是从byte数组中读取的ByteArrayInputStream byteInputStream = new ByteArrayInputStream(str.getBytes());//ByteArrayOutputStream生成对象的时候,是生成一个100大小的byte的缓冲区,写入的时候,是把内容写入内存中的一个缓冲区ByteArrayOutputStream byteOutput = new ByteArrayOutputStream(100);int i =0;byte [] b = new byte[100];while((i = byteInputStream.read(b))!= -1){byteOutput.write(b, 0, i);}System.out.println(new String(byteOutput.toByteArray()));}

打印结果是:AAAAACCCCcCBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB

把一个String字符串的内容使用ByteArrayOutputStream读取出来,然后打印显示。这个代码没有什么问题,估计大家都能写出来,但是看一下下边添加一行代码之后的内容:

@Testpublic void testByteArrayInputStream() throws Exception {String str = "AAAAACCCCcCBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB";//ByteArrayInputStream是把一个byte数组转换成一个字节流,读取的内容是从byte数组中读取的ByteArrayInputStream byteInputStream = new ByteArrayInputStream(str.getBytes());//调用这个方法,会影响到下次读取,下次再调用这个方法,读取的起始点会后移5个bytebyteInputStream.read(new byte[5]);//ByteArrayOutputStream生成对象的时候,是生成一个100大小的byte的缓冲区,写入的时候,是把内容写入内存中的一个缓冲区ByteArrayOutputStream byteOutput = new ByteArrayOutputStream(100);int i =0;byte [] b = new byte[100];while((i = byteInputStream.read(b))!= -1){byteOutput.write(b, 0, i);}System.out.println(new String(byteOutput.toByteArray()));}

打印结果是:CCCCcCBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB

我在第8行添加了一行代码,这行代码可以把String生成的byte数组中读取5个字节的内容,这行代码会影响到后边第15行byteInputStream的读取结果,显然”AAAAA“在输出中没有了,这是为什么呢?我感觉需要查看java中ByteArrayInputStream的Read方法的实现源码,查看的结果其实可以总结一个句话,就是在InputStream读取的时候,会有一个pos指针,他指示每次读取之后下一次要读取的起始位置,在API文档中是这样解释的:(看过InputStream源码的都明白,read方法其实调用的都是带有三个参数的方法)

public int read(byte[] b,int off, int len)   
 Reads up to len bytes of data into an array of bytes from this input stream. If pos equals count, then -1 is returned to indicate end of file. Otherwise, the number k of
bytes read is equal to the smaller of len and count-pos. If k is positive, then bytes buf[pos] through buf[pos+k-1] are copied into b[off] through b[off+k-1] in the manner
performed by System.arraycopy. The value k is added into pos and k is returned.

就是在每次读取的时候会更新pos的值,当你下次再来读取的时候是从pos的位置开始的,而不是从头开始,所以第二次获取String中的值的时候是不全的,”AAAAA“丢掉了,这也就导致了两次调用request.getInputStream,第二次的时候肯定获取不了值,因为第一次读取完成之后pos指针在末尾,下次再读取肯定读取不到,同request.getInputStream两次调用返回的对象是同一个对象。读取的是同一个Stream。

但是仔细查看API文档你会发现有这样一个方法:

public void reset()
Resets the buffer to the marked position. The marked position is 0 unless another position was marked or an offset was specified in the constructor. 

就是可以把pos的指针的位置重置为起始位置,但是调用它是有条件的,不是所有的IO读取流都能调用这个方法.看一下有这个方法

public boolean markSupported()
Tests if this input stream supports the mark and reset methods. Whether or not mark and reset are supported is an invariant property of a particular input stream instance. 
The markSupported method of InputStream returns false.

这个方法可以判断是不是支持reset()方法的调用,

通过查看ServletInputStream的源码,

ServletInputStream继承了InputStream同时没有重写reset()方法,查看一下InputStream源码,InputStream的reset()方法源码是这样的:

public synchronized void reset() throws IOException {throw new IOException("mark/reset not supported");}

调用reset方法直接抛出异常,所以ServletInputStream是不能调用reset方法,这就导致了只能调用一次getInputStream(),第二次调用的时候没有办法获取到InputStream流中的原因。

现在我们更改一下上边读取String字符串的例子:

 1       @Test2     public void testByteArrayInputStream() throws Exception {3         String str = "AAAAACCCCcCBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB";4         //ByteArrayInputStream是把一个byte数组转换成一个字节流,读取的内容是从byte数组中读取的5         ByteArrayInputStream byteInputStream = new ByteArrayInputStream(str.getBytes());6         7         //调用这个方法,会影响到下次读取,下次再调用这个方法,读取的起始点会后移5个byte8         byteInputStream.read(new byte[5]);9         byteInputStream.reset();//调用reset方法可以使read中的pos指针复位
10         
11         
12         //ByteArrayOutputStream生成对象的时候,是生成一个100大小的byte的缓冲区,写入的时候,是把内容写入内存中的一个缓冲区
13         ByteArrayOutputStream byteOutput = new ByteArrayOutputStream(100);
14         
15         int i =0;
16         byte [] b = new byte[100];
17         while((i = byteInputStream.read(b))!= -1){
18             byteOutput.write(b, 0, i);
19         }
20         System.out.println(new String(byteOutput.toByteArray()));
21     }

打印结果是:AAAAACCCCcCBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB

在第8行添加一行代码,byteInputStream调用reset()方法,打印的结果回复了正常。

转载于:https://www.cnblogs.com/whx7762/p/9475795.html

相关文章:

windows nodejs mysql_windows server 安装 mysql + nondejs连接mysql

下载安装下载完后,将 zip 包解压到相应的目录,这里我将解压后的文件夹放在 C:\mysql 下。接下来需要配置下 MySQL 的配置文件打开刚刚解压的文件夹 C:\mysql ,在该文件夹下创建 my.ini 配置文件,编辑 my.ini 配置以下基本信息&…

多线程并行和并发的区别

并行就是两个任务同时运行,就是甲任务进行的同时,乙任务也在进行。(需要多核CPU) 并发是指两个任务都请求运行,而处理器只能按受一个任务,就把这两个任务安排轮流进行,由于时间间隔较短,使人感觉两个任务都…

php dropdownlist,为何activitieDropdownlist的值始终无法获取到

为什么activitieDropdownlist的值始终无法获取到?视图层view控制器层public function actionDbdplist(){$m_subjectlist new SubjectList();$m_gradelist new Grade;$this->subject_list SubjectList::model()->findAll();$this->grade_list Grade::mo…

python之内置函数

一、 内置函数 什么是内置函数?就是python给你提供的可以直接使用的函数。到目前为止在python中一共有68个内置函数 经过我两个多小时的制作终于弄出了个能看的东西↓↓↓↓↓↓ 思维导图链接:https://www.processon.com/view/link/5b72a285e4b053a09c33e534 转载于…

远程桌面linux服务器配置,linux平台下远程桌面服务器的安装和设置

一、xdm 方式前提:安装linux时一定要选上xwindow,这是最基本的前提,不安装它,是绝对没有图形界面的。**********************************************************基本概念和相关命令说明:XDMCP(X Display Manager Co…

关于字符串的分割问题

1、如何快速的将一个字符串分割成一个个字符? Scanner scnew Scanner(System.in);System.out.println("请输入一个字符串:");String strsc.next();System.out.println("输入的字符串的长度为:"str.length());char [] ast…

java创建对象_java 创建对象的五种方式

通过 Class 对象的 getConstructor 可以获取 java.lang.reflect.Constructor 对象Constructor 对象用来描述类的构造方法,通过给 getConstructor 方法传入不同的 Class 对象,可以获取到对应的无参或有参数的构造方法通过 Constructor 的 newInstance 方法…

php拍照从手机相册中选择,微信js-sdk预览图片接口及从拍照或手机相册中选图接口用法示例...

本文实例讲述了微信js-sdk预览图片接口及从拍照或手机相册中选图接口用法。分享给大家供大家参考,具体如下:目前中js-sdk 1.0版本中,预览图片提供了2个接口,接口的定义参考官方文档1.预览网络图片http链接的2.预览本地图片wenxin:…

BZOJ 1124: [POI2008]枪战Maf(构造 + 贪心)

题意 有 \(n\) 个人,每个人手里有一把手枪。一开始所有人都选定一个人瞄准(有可能瞄准自己)。然后他们按某个顺序开枪,且任意时刻只有一个人开枪。 因此,对于不同的开枪顺序,最后死的人也不同。 问最后死的…

Maven跳过测试

Maven跳过测试用例 在properties中声明<properties><maven.test.skip>true</maven.test.skip> </properties> 或者 <properties><skipTests>true</skipTests> </properties> 在执行命令中声明mvn test -Dmaven.test.skiptrue …

Linux内核 题目,《Linux内核完全注释》部分习题答案

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼第3章 内核引导和启动过程2.为什么不直接将system模块搬到0x00000处而是先搬到0x10000处&#xff0c;再搬到0x00000处呢&#xff1f;在机器开机上电时&#xff0c;ROM BIOS将bootsect代码加载到内存的固定位置0x7c00处&#xff0c;…

java jdk 1.8 安装_下载、安装、配置 java jdk1.8

近期配置react native的开发环境&#xff0c;所以就从配置环境开始。rn的环境配置有那么几项&#xff0c;其中重要的一个就是java jdk(Java Development Kit 的缩写)&#xff0c;那么以下就是下载、安装还有配置的流程1.下载java jdk 1.8在地址栏输入 java jdk,如下图所示&…

liunx php redis扩展,CentOS 7下安装php-redis扩展及简单使用

前言&#xff1a;在本篇文章中&#xff0c;我将给大家介绍如何在CentOS7上安装PHP-Redis扩展以及一些简单的实用&#xff0c;关于如何在Centos上安装redis的&#xff0c;可以参考想要在php中操作redis&#xff0c;那就必须安装php-redis扩展&#xff0c;就比如MySQL一样&#x…

Luogu 2470 [SCOI2007]压缩

和Luogu 4302 [SCOI2003]字符串折叠 差不多的想法&#xff0c;区间dp 为了计算方便&#xff0c;我们可以假设区间[l, r]的前面放了一个M&#xff0c;设$f_{i, j, 0/1}$表示区间$[i, j]$中是否存在M 因为这题只能是二的幂次倍压缩&#xff0c;所以转移的时候枚举中点chk是否合法…

做图形处理Linux小型主机,8个优秀的linux图形图像工具

对艺术家、摄影师、动画师和设计师而言&#xff0c;Linux是一个有潜力的平台。廉价的硬件&#xff0c;优秀的免费软件&#xff0c;任何有才华的人都能在上面创作专业水平的计算机图形。开源社区提供了丰富的开源图形工具&#xff0c;但要慧眼识珠并非易事。这里介绍的优秀图形工…

使用laravel框架的eloquent\DB模型连接多个数据库

1、配置.env文件 DB_HOST_TRAILER127.0.0.1DB_PORT_TRAILER3306DB_DATABASE_TRAILERhtms_trailerDB_USERNAME_TRAILERrootDB_PASSWORD_TRAILER DB_HOST_FREIGHT127.0.0.1DB_PORT_FREIGHT3306DB_DATABASE_FREIGHThangli_saasDB_USERNAME_FREIGHTrootDB_PASSWORD_FREIGHT 2、配置…

java openfile busy_android java.io.IOException: open failed: EBUSY (Device or resource busy)

今天遇到一个奇怪的问题&#xff0c;测试在程序的下载界面&#xff0c;下载一个文件第一次下载成功&#xff0c;删除后再下载结果下载报错&#xff0c;程序&#xff1a;file.createNewFile();报错&#xff1a;java.io.IOException: open failed: EBUSY (Device or resource bus…

java service注入失败,使用spring向service里面注入dao不成功。

使用spring向service里面注入dao不成功。求救啊&#xff01;本帖最后由 PaperStar 于 2013-12-26 19:29:20 编辑页面调用action&#xff0c;action调用service&#xff0c;service调用dao用Debug查看action调用service方法时service有值&#xff0c;但是service调用dao时&#…

下面为初学者分享一下SQL 数据库学习资料

一、基础1、说明&#xff1a;创建数据库CREATE DATABASE database-name2、说明&#xff1a;删除数据库drop database dbname3、说明&#xff1a;备份sql server--- 创建 备份数据的 deviceUSE masterEXEC sp_addumpdevice disk, testBack, c:\mssql7backup\MyNwind_1.dat--- 开…

linux7设置时间,CentOS 7 设置日期和时间

现代操作系统分为以下两种类型的时钟&#xff1a;实时时钟(Real-Time Clock&#xff0c;RTC)&#xff0c;通常称为硬件时钟(一般是系统主板上的集成电路)&#xff0c;它完全独立于操作系统的当前状态&#xff0c;即使在计算机关闭时也能运行。系统时钟&#xff0c;也称为软件时…

SQLMap安装步骤

SQLMap是利用Python语言写的&#xff0c;所以需要将Python这个语言环境给安装上 &#xff1a; 1、首先下载Python(这里Python版本为2.7.2&#xff0c;可以下载不同或高版本的) 2、然后在下载sqlmap&#xff08;http://sqlmap.org&#xff09; 3、这两个软件下载完成后&#xff…

am5718_AM5718如何扩大内存 - Sitara™ Cortex-A8 和 ARM9 微处理器 - Sitara™ Cortex-A8 和 ARM9 微处理器 - E2E™ 中文支持论坛...

谢谢了Shine,你的资料和建议非常到位&#xff0c;按您的建议&#xff0c;修改了board.c以下两处&#xff0c;问题解决了。1&#xff1a;board/ti/am57xx/board.c文件static const struct dmm_lisa_map_regs am571x_idk_lisa_regs {.dmm_lisa_map_3 0x80640100,.is_ma_present…

亚马逊刊登php代码,最全的亚马逊刊登listing工具了解一下

如果你是亚马逊FBA卖家&#xff0c;那么你可能会错过很多有用的亚马逊listing工具。这些listing工具可以批量上传listing&#xff0c;同时还可以记录产品特征&#xff0c;以及打印运输标签。1、易仓刊登系统易仓刊登系统是一款易仓基于已有ERP客户需求研发的一套平台产品刊登系…

linux重命名tar命令,linux常用操作指令4 —— 文件操作相关命令(mkdir、touch、rm、mv、cp、cat 、 find 、tar、chmod)...

文件操作相关命令文件操作相关命令1、创建文件夹mkdir2、创建文件touch3、移动文件夹mv(类似于剪切)4、删除rm5、重命名mv6、复制cp7、查看文件(cat、head、tail..)8、查找文件 find (重要)9、归档压缩tar10、修改文件权限chmod参考文件操作相关命令1、创建文件夹mkdir# mkdir …

后台生成小程序码

工作需要&#xff0c;根据动态参数生成小程序二维码。 找了下开发API &#xff1a;https://developers.weixin.qq.com/miniprogram/dev/api/qrcode.html 选择了B接口&#xff0c;可以无限生成&#xff0c;只是参数有点限制&#xff0c;但是可以满足需求&#xff0c;开搞。 一、…

2017-02-20 注册.Net Framework4.0

在使用IIS发布Web应用程序时&#xff0c;有时会遇到Asp.Net 4.0尚未在Web服务器上注册的问题&#xff0c;需要手动注册下.Net Framework 4.0。 注册.net Framwork4.0 步骤&#xff0c;以windows7系统为例&#xff0c;注册 步骤如下&#xff1a; 64位操作系统&#xff1a; 1. …

java字符存储,在什么编码是Java字符存储在?

Is the Java char type guaranteed to be stored in any particular encoding?Edit: I phrased this question incorrectly. What I meant to ask is are char literals guaranteed to use any particular encoding?解决方案"Stored" where? All Strings in Java …

matlab 仿真步长,MATLAB Simulink变步长仿真与固定步长仿真简单对比

今天晚上翻了一下资料发现&#xff0c;关于变步长以及固定步长仿真的理解我之前是由错误理解的。当时没有做什么认真的思考活着尝试就自己给自己下了一个结论&#xff1a;变步长仿真会比较精确&#xff0c;但是可能会消耗更多的计算机资源&#xff01;错&#xff01;大错特错&a…

JS设计模式(13)状态模式

什么是状态模式&#xff1f; 定义&#xff1a;将事物内部的每个状态分别封装成类&#xff0c;内部状态改变会产生不同行为。 主要解决&#xff1a;对象的行为依赖于它的状态&#xff08;属性&#xff09;&#xff0c;并且可以根据它的状态改变而改变它的相关行为。 何时使用&am…

【转】Mac 程序员的十种武器

http://chijianqiang.baijia.baidu.com/article/3733 上 在写 Mac 程序员的十个武器之前&#xff0c;我决定先讲一个故事&#xff0c;关于 Mac 和爱情的。&#xff08;你们不是问 Mac 和爱情有个鸟关系吗&#xff1f;&#xff09; 从前有一个孩子叫做小明&#xff0c;他不是高帅…