您的位置:首页 > 编程语言 > Java开发

Java:IO流之字节流InputStream、OutputStream详解

2015-10-28 16:50 537 查看
字节流:
(抽象基类)InputStream类(读):
(抽象基类)OutputStream类(写):

InputStream:
构造方法摘要
InputStream()

方法摘要
int available()
返回此输入流下一个方法调用可以不受阻塞地从此输入流读取(或跳过)的估计字节数。
void close()
关闭此输入流并释放与该流关联的所有系统资源。
void mark(int readlimit)
在此输入流中标记当前的位置。
boolean markSupported()
测试此输入流是否支持 mark 和 reset 方法。
abstract int read()
从输入流中读取数据的下一个字节。
int read(byte[] b)
从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b 中。
int read(byte[] b, int off, int len)
将输入流中最多 len 个数据字节读入 byte 数组。
void reset()
将此流重新定位到最后一次对此输入流调用 mark 方法时的位置。
long skip(long n)
跳过和丢弃此输入流中数据的 n 个字节。
//例子1:

import java.io.*;
class FileInputStreamDemo
{
public static void main(String[] args)throws IOException
{
//读取该文件中的数据
FileInputStream fis = new FileInputStream("f:\\myfile\\fos.txt");
int b = 0;
while((b = fis.read())!=-1)
{
System.out.print((char)b);
}

//此处不适用于刷新,而是用于关闭资源
fis.close();
}
}


OutputStream:
构造方法摘要
OutputStream()

方法摘要
void close()
关闭此输出流并释放与此流有关的所有系统资源。
void flush()
刷新此输出流并强制写出所有缓冲的输出字节。
void write(byte[] b)
将 b.length 个字节从指定的 byte 数组写入此输出流。
void write(byte[] b, int off, int len)
将指定 byte 数组中从偏移量 off 开始的 len 个字节写入此输出流。
abstract void write(int b)
将指定的字节写入此输出流。

//例子2:需求:想要操作图片数据,这时就要用到字节流。

import java.io.*;
class FileOutputStreamDemo
{
public static void main(String[] args)throws IOException
{
//创建字节流文件并写入数据
FileOutputStream fos = new FileOutputStream("f:\\myfile\\fos.txt");
String str = "abcde";
byte b[] = str.getBytes();
fos.write(b);

//此处不适用于刷新,而是用于关闭资源
fos.close();
}
}


使用字节流读取文件中的数据有三种方式:
方法一:一个一个字节的读取;
方法二:先指定长度的数组,再将数据读读入数组,最后从数组中读出数据内容;
方法三:通过available()方法可以获取文件中数据的总个数,以此个数设置为数组长度刚刚好,剩下步骤与方法二相等。
//例子3:

import java.io.*;
class FileStream
{
public static void main(String[] args)throws IOException
{
//WriteFile();
ReadFile1();
ReadFile2();
ReadFile3();
}
public static void WriteFile()throws IOException
{
//创建字节流文件并写入数据
FileOutputStream fos = new FileOutputStream("f:\\myfile\\fs.txt");
String str = "xyzabcd123";
byte b[] = str.getBytes();
fos.write(b);

//此处不适用于刷新,而是用于关闭资源
fos.close();
}

//方法一:一个一个字节的读取
public static void ReadFile1()throws IOException
{
//读取该文件中的数据
FileInputStream fis = new FileInputStream("f:\\myfile\\fs.txt");
int b = 0;
while((b = fis.read())!=-1)
{
System.out.println((char)b);
}

//此处不适用于刷新,而是用于关闭资源
fis.close();
}

//方法二:先指定长度的数组,再将数据读读入数组,最后从数组中读出数据内容。
public static void ReadFile2()throws IOException
{
//读取该文件中的数据
FileInputStream fis = new FileInputStream("f:\\myfile\\fs.txt");
byte[] buf = new byte[1024];
int num = 0;
while((num = fis.read(buf))!=-1)
{
System.out.println(new String(buf,0,num));
}

//此处不适用于刷新,而是用于关闭资源
fis.close();
}

//方法三:通过available()方法可以获取文件中数据的总个数,以此个数设置为数组长度刚刚好,剩下步骤与方法二相同。
public static void ReadFile3()throws IOException
{
//读取该文件中的数据
FileInputStream fis = new FileInputStream("f:\\myfile\\fs.txt");
byte[] buf = new byte[fis.available()];
fis.read(buf);
System.out.println(new String(buf));
//此处不适用于刷新,而是用于关闭资源
fis.close();
}
}


复制和读取二进制文件,必须使用字节流,下面进行具体的应用实例举例:
//复制图片:

思路:
1、用字节读取流对象和图片关联;
2、用字节写入流对象创建一个图片文件,用于存储获取到的数据;
3、通过循环读写,完成数据的存储;
4、关闭资源。

import java.io.*;
class CopyPicture
{
public static void main(String[] args)
{
FileOutputStream fos = null;
FileInputStream fis = null;
try
{
fos = new FileOutputStream("F:\\myfile\\2.jpg");
fis = new FileInputStream("F:\\myfile\\1.jpg");
byte[] buf = new byte[1024];
int len = 0;
while((len = fis.read(buf))!=-1)
{
fos.write(buf,0,len);
}
}
catch(IOException e)
{
throw new RuntimeException("文件复制失败!");
}
finally
{
try
{
if(fis!=null)
fis.close();
}
catch(IOException e)
{
throw new RuntimeException("读取流关闭异常!");
}
try
{
if(fos!=null)
fos.close();
}
catch(IOException e)
{
throw new RuntimeException("写入流关闭异常!");
}
}
}
}


//复制MP3:

复制一个音频。(通过Buffered缓冲空间)
BufferedInputStream
BufferedOutputStream
思路:
1、用字节读取流对象和音频关联;
2、用字节写入流对象创建一个音频文件,用于存储获取到的数据;
3、通过循环读写,完成数据的存储;
4、关闭资源。

import java.io.*;
class CopyMp3
{
public static void main(String[] args)
{
long start = System.currentTimeMillis();
copy();
long end = System.currentTimeMillis();
System.out.println((end-start)+"毫秒");
}

//通过字节流缓冲区完成复制
public static void copy()
{
BufferedInputStream bufis = null;
BufferedOutputStream bufos = null;
try
{
bufis = new BufferedInputStream(new FileInputStream("F:\\myfile\\b01.mp3"));
bufos = new BufferedOutputStream(new FileOutputStream("F:\\myfile\\b02.mp3"));
int by = 0;
while((by = bufis.read())!=-1)
{
bufos.write(by);
}
}
catch(IOException e)
{
throw new RuntimeException("文件复制失败!");
}
finally
{
try
{
if(bufis!=null)
bufis.close();
}
catch(IOException e)
{
throw new RuntimeException("读取流关闭异常!");
}
try
{
if(bufos!=null)
bufos.close();
}
catch(IOException e)
{
throw new RuntimeException("写入流关闭异常!");
}
}
}
}


//复制视频avi:

复制一个视频。(通过自定义的数组缓冲空间)
思路:
1、用字节读取流对象和视频关联;
2、用字节写入流对象创建一个视频文件,用于存储获取到的数据;
3、通过循环读写,完成数据的存储;
4、关闭资源。

import java.io.*;
class CopyAvi
{
public static void main(String[] args)
{
FileOutputStream fos = null;
FileInputStream fis = null;
try
{
fos = new FileOutputStream("F:\\myfile\\a02.avi");
fis = new FileInputStream("F:\\myfile\\a01.avi");
byte[] buf = new byte[4096];
int len = 0;
while((len = fis.read(buf))!=-1)
{
fos.write(buf,0,len);
}
}
catch(IOException e)
{
throw new RuntimeException("文件复制失败!");
}
finally
{
try
{
if(fis!=null)
fis.close();
}
catch(IOException e)
{
throw new RuntimeException("读取流关闭异常!");
}
try
{
if(fos!=null)
fos.close();
}
catch(IOException e)
{
throw new RuntimeException("写入流关闭异常!");
}
}
}
}


最后再提一下,使用api中的字节流是很好的,当然,用户也可以自定义一个字节流缓冲区。下面就给一个具体的实例:
//演示mp3的复制,自定义一个字节流缓冲区

import java.io.*;
class MyBufferedStream
{
private InputStream in;
private byte[] buf = new byte[1024];
private int pos = 0,count = 0;
MyBufferedStream(InputStream in)
{
this.in = in;
}
//一次读一个字节,从缓冲区(字节数组)获取
public int myread() throws IOException
{
//通过in对象来读取硬盘上的字节数据,存放在字节数组中。
if(count==0)
{
count  = in.read(buf);
if(count<0)
return -1;
pos=0;
byte b = buf[pos];
count--;
pos++;
return b&255;//byte字节提升为整型int型
}
else if(count>0)
{
byte b = buf[pos];
count--;
pos++;
return b&0xff;//byte字节提升为整型int型+-++++
}
return -1;
}
public void myclose()throws IOException
{
in.close();
}
}
class MyBufferedStreamDemo
{
public static void main(String[] args)throws IOException
{
long start = System.currentTimeMillis();
copymp3();
long end = System.currentTimeMillis();
System.out.println((end-start)+"毫秒");
}
public static void copymp3()throws IOException
{
MyBufferedStream mys  = new MyBufferedStream(new FileInputStream("F:\\myfile\\b01.mp3"));
BufferedOutputStream bufos = new BufferedOutputStream(new FileOutputStream("F:\\myfile\\b03.mp3"));
int num = 0;
//System.out.println("第一个字节是:"+mys.myread());
while((mys.myread())!=-1)//要区别开-1是数据读完时的-1,还是读取的字节数据内容为-1,所以要用&操作.
{
bufos.write(num);
}
mys.myclose();
bufos.close();
}
}


提升
byte:-1 -------> int:-1
11111111 11111111-11111111-11111111-111111111
11111111---->提升了一个int类型,那不还是-1吗?是-1的原因是是因为在8个1前面补的1导致的。
那么,我只要在8个1的前面补充0,即可以保持原字节数不变,又可以避免-1的出现。
11111111-11111111-11111111-11111111
& 00000000-00000000-00000000-11111111
------------------------------------
00000000-00000000-00000000-11111111
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: