ByteArrayInputStream简介和使用_动力节点Java学院整理
ByteArrayInputStream介绍
ByteArrayInputStream是字节数组输入流。它继承于InputStream。
它包含一个内部缓冲区,该缓冲区包含从流中读取的字节;通俗点说,它的内部缓冲区就是一个字节数组,而ByteArrayInputStream本质就是通过字节数组来实现的。
我们都知道,InputStream通过read()向外提供接口,供它们来读取字节数据;而ByteArrayInputStream的内部额外的定义了一个计数器,它被用来跟踪read()方法要读取的下一个字节。
InputStream函数列表
//构造函数 InputStream() intavailable() voidclose() voidmark(intreadlimit) booleanmarkSupported() intread(byte[]buffer) abstractintread() intread(byte[]buffer,intoffset,intlength) synchronizedvoidreset() longskip(longbyteCount)
ByteArrayInputStream函数列表
//构造函数 ByteArrayInputStream(byte[]buf) ByteArrayInputStream(byte[]buf,intoffset,intlength) synchronizedintavailable() voidclose() synchronizedvoidmark(intreadlimit) booleanmarkSupported() synchronizedintread() synchronizedintread(byte[]buffer,intoffset,intlength) synchronizedvoidreset() synchronizedlongskip(longbyteCount)
InputStream和ByteArrayInputStream源码分析
InputStream是ByteArrayInputStream的父类,我们先看看InputStream的源码,然后再学ByteArrayInputStream的源码。
1.InputStream.java源码分析(基于jdk1.7.40)
packagejava.io; publicabstractclassInputStreamimplementsCloseable{ //能skip的大小 privatestaticfinalintMAX_SKIP_BUFFER_SIZE=; //从输入流中读取数据的下一个字节。 publicabstractintread()throwsIOException; //将数据从输入流读入byte数组。 publicintread(byteb[])throwsIOException{ returnread(b,0,b.length); } //将最多len个数据字节从此输入流读入byte数组。 publicintread(byteb[],intoff,intlen)throwsIOException{ if(b==null){ thrownewNullPointerException(); }elseif(off<0||len<0||len>b.length-off){ thrownewIndexOutOfBoundsException(); }elseif(len==0){ return0; } intc=read(); if(c==-1){ return-1; } b[off]=(byte)c; inti=1; try{ for(;i0){ nr=read(skipBuffer,0,(int)Math.min(size,remaining)); if(nr<0){ break; } remaining-=nr; } returnn-remaining; } publicintavailable()throwsIOException{ return0; } publicvoidclose()throwsIOException{} publicsynchronizedvoidmark(intreadlimit){} publicsynchronizedvoidreset()throwsIOException{ thrownewIOException("mark/resetnotsupported"); } publicbooleanmarkSupported(){ returnfalse; } }
2.ByteArrayInputStream.java源码分析(基于jdk1.7.40)
packagejava.io; publicclassByteArrayInputStreamextendsInputStream{ //保存字节输入流数据的字节数组 protectedbytebuf[]; //下一个会被读取的字节的索引 protectedintpos; //标记的索引 protectedintmark=0; //字节流的长度 protectedintcount; //构造函数:创建一个内容为buf的字节流 publicByteArrayInputStream(bytebuf[]){ //初始化“字节流对应的字节数组为buf” this.buf=buf; //初始化“下一个要被读取的字节索引号为” this.pos=; //初始化“字节流的长度为buf的长度” this.count=buf.length; } //构造函数:创建一个内容为buf的字节流,并且是从offset开始读取数据,读取的长度为length publicByteArrayInputStream(bytebuf[],intoffset,intlength){ //初始化“字节流对应的字节数组为buf” this.buf=buf; //初始化“下一个要被读取的字节索引号为offset” this.pos=offset; //初始化“字节流的长度” this.count=Math.min(offset+length,buf.length); //初始化“标记的字节流读取位置” this.mark=offset; } //读取下一个字节 publicsynchronizedintread(){ return(posb.length-off){ thrownewIndexOutOfBoundsException(); } if(pos>=count){ return-1; } intavail=count-pos; if(len>avail){ len=avail; } if(len<=0){ return0; } System.arraycopy(buf,pos,b,off,len); pos+=len; returnlen; } //跳过“字节流”中的n个字节。 publicsynchronizedlongskip(longn){ longk=count-pos; if(n 说明:
ByteArrayInputStream实际上是通过“字节数组”去保存数据。
(01)通过ByteArrayInputStream(bytebuf[])或ByteArrayInputStream(bytebuf[],intoffset,intlength),我们可以根据buf数组来创建字节流对象。
(02)read()的作用是从字节流中“读取下一个字节”。
(03)read(byte[]buffer,intoffset,intlength)的作用是从字节流读取字节数据,并写入到字节数组buffer中。offset是将字节写入到buffer的起始位置,length是写入的字节的长度。
(04)markSupported()是判断字节流是否支持“标记功能”。它一直返回true。
(05)mark(intreadlimit)的作用是记录标记位置。记录标记位置之后,某一时刻调用reset()则将“字节流下一个被读取的位置”重置到“mark(intreadlimit)所标记的位置”;也就是说,reset()之后再读取字节流时,是从mark(intreadlimit)所标记的位置开始读取。
示例代码
关于ByteArrayInputStream中API的详细用法,参考示例代码(ByteArrayInputStreamTest.java):
importjava.io.ByteArrayInputStream; importjava.io.ByteArrayOutputStream; /** *ByteArrayInputStream测试程序 * */ publicclassByteArrayInputStreamTest{ privatestaticfinalintLEN=5; //对应英文字母“abcddefghijklmnopqrsttuvwxyz” privatestaticfinalbyte[]ArrayLetters={ 0x61,0x62,0x63,0x64,0x65,0x66,0x67,0x68,0x69,0x6A,0x6B,0x6C,0x6D,0x6E,0x6F, 0x70,0x71,0x72,0x73,0x74,0x75,0x76,0x77,0x78,0x79,0x7A }; publicstaticvoidmain(String[]args){ Stringtmp=newString(ArrayLetters); System.out.println("ArrayLetters="+tmp); tesByteArrayInputStream(); } /** *ByteArrayInputStream的API测试函数 */ privatestaticvoidtesByteArrayInputStream(){ //创建ByteArrayInputStream字节流,内容是ArrayLetters数组 ByteArrayInputStreambais=newByteArrayInputStream(ArrayLetters); //从字节流中读取5个字节 for(inti=0;i=0){ //读取“字节流的下一个字节” inttmp=bais.read(); System.out.printf("%d:0x%s\n",i,Integer.toHexString(tmp)); } } //若“该字节流”不支持标记功能,则直接退出 if(!bais.markSupported()){ System.out.println("makenotsupported!"); return; } //标记“字节流中下一个被读取的位置”。即--标记“0x66”,因为因为前面已经读取了5个字节,所以下一个被读取的位置是第6个字节” //(01),ByteArrayInputStream类的mark(0)函数中的“参数0”是没有实际意义的。 //(02),mark()与reset()是配套的,reset()会将“字节流中下一个被读取的位置”重置为“mark()中所保存的位置” bais.mark(0); //跳过5个字节。跳过5个字节后,字节流中下一个被读取的值应该是“0x6B”。 bais.skip(5); //从字节流中读取5个数据。即读取“0x6B,0x6C,0x6D,0x6E,0x6F” byte[]buf=newbyte[LEN]; bais.read(buf,0,LEN); //将buf转换为String字符串。“0x6B,0x6C,0x6D,0x6E,0x6F”对应字符是“klmno” Stringstr1=newString(buf); System.out.printf("str1=%s\n",str1); //重置“字节流”:即,将“字节流中下一个被读取的位置”重置到“mark()所标记的位置”,即0x66。 bais.reset(); //从“重置后的字节流”中读取5个字节到buf中。即读取“0x66,0x67,0x68,0x69,0x6A” bais.read(buf,0,LEN); //将buf转换为String字符串。“0x66,0x67,0x68,0x69,0x6A”对应字符是“fghij” Stringstr2=newString(buf); System.out.printf("str2=%s\n",str2); } } 运行结果:
ArrayLetters=abcdefghijklmnopqrstuvwxyz 0:0x61 1:0x62 2:0x63 3:0x64 4:0x65 str1=klmno str2=fghij结果说明:
(01)ArrayLetters是字节数组。0x61对应的ASCII码值是a,0x62对应的ASCII码值是b,依次类推...
(02)ByteArrayInputStreambais=newByteArrayInputStream(ArrayLetters);这句话是创建“字节流bais”,它的内容就是ArrayLetters。
(03)for(inti=0;i
(04)bais.mark(0);这句话就是“设置字节流的标记”,此时标记的位置对应的值是0x66。
(05)bais.skip(5);这句话是跳过5个字节。跳过5个字节后,对应的字节流中下一个被读取的字节的值是0x6B。
(06)bais.read(buf,0,LEN);这句话是“从字节流中读取LEN个数据写入到buf中,0表示从buf的第0个位置开始写入”。
(07)bais.reset();这句话是将“字节流中下一个被读取的位置”重置到“mark()所标记的位置”,即0x66。
学完了ByteArrayInputStream输入流。下面,我们学习与之对应的输出流ByteArrayOutputStream。
以上所述是小编给大家介绍的ByteArrayInputStream简介和使用,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对毛票票网站的支持!