博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java:IO流之字节流InputStream、OutputStream详解
阅读量:5136 次
发布时间:2019-06-13

本文共 8882 字,大约阅读时间需要 29 分钟。

字节流:
(抽象基类)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
 

转载于:https://www.cnblogs.com/XYQ-208910/p/4917818.html

你可能感兴趣的文章
在iOS开发中,给项目添加新的.framework
查看>>
推荐18个基于 HTML5 Canvas 开发的图表库
查看>>
ACM/ICPC 之 优先级队列+设置IO缓存区(TSH OJ-Schedule(任务调度))
查看>>
ACM/ICPC 之 分治法入门(画图模拟:POJ 2083)
查看>>
php session设置说明 ,PHP中session_unset与session_destroy的区别 (转)
查看>>
android键盘事件
查看>>
网络编程- 解决黏包现象方案二之struct模块(七)
查看>>
这是最好的时光 这是最坏的时光 v0.1.1.1
查看>>
(三)微信小程序之发送服务通知(模板消息)
查看>>
linux命令ll -d
查看>>
[转载]ios中摄像头/相册获取图片,压缩图片,上传服务器方法总结
查看>>
用面向对象的思想设计问题
查看>>
Git Stash用法
查看>>
变量指针作为模板参数
查看>>
QComboBox用法小列(转)
查看>>
Leetcode 19. Remove Nth Node From End of List(python)
查看>>
WPF x:static的使用
查看>>
iOS Aspect Fit,Aspect Fill,Scale To Fill
查看>>
安装xhprof和graphviz
查看>>
c# 操作XML
查看>>