在使用io操作之前,先看一下java中的文件类File如何使用。File包括文件和目录,对文件和目录的操作是新建目录mkdir,新建文件createNewFile,删除文件和目录delete,以及其他的一些操作。

package zaojiahua;

import java.io.File;
import java.io.IOException; /* 对文件类的操作 */
public class Test
{
public static void main(String[] args) throws IOException
{
//新建一个目录
File file = new File("1/2");
//判断目录是否存在
if(!file.exists())
{
//创建多层目录
file.mkdirs();
} //创建文件
File file2 = new File(file,"1.txt");
file2.createNewFile(); //创建文件
File file3 = new File("2.txt");
file3.createNewFile();
//获取文件的绝对路径
System.out.println(file3.getAbsolutePath());
System.out.println(file2.getParent());
//删除目录
file.delete(); File file4 = new File("1/3");
//遍历目录中的文件和目录,是单层遍历
String str[] = file4.list();
for(String s : str)
{
System.out.println(s);
} //列出一个目录下的所有文件
System.out.println("**********************listAll**********************");
listAll(file4);
} static void listAll(File filename)
{
if(filename.isDirectory())
{
File fileDir [] = filename.listFiles();
for(File file : fileDir)
{
if(file.isDirectory())
{
System.out.println("目录:"+file);
listAll(file);
}
else
System.out.println("文件:"+file);
}
}
else
{
System.out.println("文件:"+filename);
}
} }

接下来是java中IO类的操作,IO分类如下图所示,这里只说几个常用的。

首先是FileInputStream和FileOutputStream,这一对流是对文件进行读写的,读写方法是从父类inputStream和OutputStream继承下来的read和write方法。

package com.zaojiahua.iodemo;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException; public class Test { public static void main(String[] args) throws IOException { File file = new File("file.txt");
//实现数据的拷贝
FileInputStream inputStream = new FileInputStream(file);
FileOutputStream outputStream = new FileOutputStream("1.txt"); // int i;
// while((i = inputStream.read() ) != -1)
// {
// //read的返回值是int,如果遇到了文件尾则返回-1
// outputStream.write(i);
// } //或者使用如下的方法
byte [] buf = new byte[1024];
int len = -1;
while((len = inputStream.read(buf)) != -1)
{
outputStream.write(buf);
} inputStream.close();
outputStream.close();
} }

BufferedInputStream对外提供滑动读取的功能实现,通过预先读入一整段原始输入流数据至缓冲区中,而外界对BufferedInputStream的读取操作实际上是在缓冲区上进行,如果读取的数据超过了缓冲区的范围,那么BufferedInputStream负责重新从原始输入流中载入下一截数据填充缓冲区,然后外界继续通过缓冲区进行数据读取。这样的设计的好处是:避免了大量的磁盘IO,因为原始的InputStream类实现的read是即时读取的,即每一次读取都会是一次磁盘IO操作(哪怕只读取了1个字节的数据),可想而知,如果数据量巨大,这样的磁盘消耗非常可怕。而通过缓冲区的实现,读取可以读取缓冲区中的内容,当读取超过缓冲区的内容后再进行一次磁盘IO,载入一段数据填充缓冲,那么下一次读取一般情况下就直接可以从缓冲区读取,减少了磁盘IO。说白了buffered就是用来缓存的,可以用来提高读取的效率,之所以说FileInputStream是阻塞的方法是因为CUP的速度和磁盘的速度是不匹配的,如果每次要读取的时候都访问磁盘这样就造成了阻塞。通过以上的说明可以看出buffered则并不是阻塞的。所以我们读取文件的时候一般都在文件流上边套上一层buffer流。

package com.zaojiahua.iodemo;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException; public class Test { public static void main(String[] args) throws IOException { File file = new File("file.txt");
//使用buffer实现数据的拷贝
BufferedInputStream input = new BufferedInputStream(new FileInputStream(file));
BufferedOutputStream output = new BufferedOutputStream(new FileOutputStream("1.txt")); //或者使用如下的方法
byte [] buf = new byte[1024];
int len = -1;
while((len = input.read(buf)) != -1)
{
output.write(buf,0,len);
//将读取到的字节数据转化为字符串打印出来
String str = new String(buf,0,len);
System.out.println(str);
} //关闭流
input.close();
output.close();
} }

数据在硬盘上都是以字节的方式存储的,那么什么时候使用字符流,什么时候使用字节流呢?字符流是对字符操作的,也就是对文本文件或者其他字符文件操作。而对于图片,声音,视频这些文件则用字节流操作。接下来是对字符流的操作,字符流相当于是字节流+编码表。可以向文件中直接写入字符串。读取的时候注意是用字符数组接受,不是字节数组。

package com.zaojiahua.iodemo;

import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException; public class Test { public static void main(String[] args) throws IOException { File file = new File("file.txt"); //使用字符流实现文本文件的复制
FileReader reader = new FileReader(file);
FileWriter writer = new FileWriter("2.txt");
int len;
//这里是字符数组,不是字节数组
char [] buf = new char[1024];
while((len = reader.read(buf)) != -1)
{
System.out.println(new String(buf,0,len));
writer.write(buf,0,len);
}
//可以将字符串直接写到文件中
writer.write("\r\n"+"end"); //关闭流
reader.close();
writer.close();
} }

以下是缓冲的字符流。经过缓冲以后可以读取和写入一行的数据。

package com.zaojiahua.iodemo;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException; public class Test { public static void main(String[] args) throws IOException { File file = new File("file.txt"); //使用字符流实现文本文件的复制
BufferedReader bufferedReader = new BufferedReader(new FileReader(file));
BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter("2.txt")); String buf;
//使用buffer缓冲的时候可以读取和写入一行
while((buf = bufferedReader.readLine()) != null)
{
bufferedWriter.write(buf);
//写入一行
bufferedWriter.newLine();
System.out.println(buf);
} //关闭流
bufferedReader.close();
bufferedWriter.close();
}
}

最后解决的一个问题是字节流和字符流的转化,使用的是InputStreamReader和OutputStreamWriter,它们本身属于的是reader和writer字符流,我们之所以会用到这些转化流是因为系统有时候只给我们提供了字节流,为了方便操作,要用到字符流。比如说System.in标准输入流就是字节流。你想从那里得到用户在键盘上的输入,只能是以转换流将它转换为Reader以方便自己的程序读取输入。再比如说Socket里的getInputStream()很明显只给你提供字节流,你要想读取字符,就得给他套个InputStreamReader()用来读取。

package com.zaojiahua.iodemo;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter; public class Test { public static void main(String[] args) throws IOException { //字节流和字符流的相互转化
FileInputStream fileInputStream = new FileInputStream("input.txt");
//inputSreamReader本来就是reader对象,创建的时候需要传入一个InputStream对象,将字节流转化为字符流
BufferedReader reader = new BufferedReader(new InputStreamReader(fileInputStream)); //将字符流转化为字节流
OutputStreamWriter outputStreamWriter = new OutputStreamWriter(new FileOutputStream("output.txt"));
BufferedWriter writer = new BufferedWriter(outputStreamWriter); //实现拷贝文件的操作
String buf;
while((buf = reader.readLine()) != null)
{
writer.write(buf);
writer.newLine();
System.out.println(buf);
} //关闭流
reader.close();
writer.close();
}
}

最新文章

  1. [原] KVM虚拟机网络闪断分析
  2. VS2010+64+OSG3.2.1之五Plugins dae编译
  3. 编译内核实现iptables防火墙layer7应用层过滤 (三)
  4. Linux 下新增虚拟内存
  5. iOS调用HTML
  6. 一次非常有意思的 SQL 优化经历
  7. Docker日志自动化: ElasticSearch、Logstash、Kibana以及Logspout
  8. 一淘搜索网页抓取系统的分析与实现(3)—scrapy+webkit & mysql+django
  9. SQLite数据库查看工具(免费)
  10. Java基础入门知识
  11. web-iPhone X
  12. hibernate第二天
  13. JS-随机生成的密码
  14. redis.conf常用配置说明
  15. Oarcle 之连接查询
  16. Python3 tkinter基础 Tk quit 点击按钮退出窗体
  17. face recognition[variations of softmax][L-Softmax]
  18. CAS5.X 集群配置 初版
  19. 欲善其工必先利其器-----ThinkPad E430加装SSD固态硬盘和内存
  20. 网络之OSI七层协议模型、TCP/IP四层模型

热门文章

  1. 关于python2.7的md5加密遇到的问题(TypeError: Unicode-objects must be encoded before hashing)
  2. java数据结构和算法04(链表)
  3. html5改良的input元素的种类
  4. Java:核心概念j积累(一)
  5. Sass的的使用二
  6. mvc框架 与vuex的介绍
  7. 通过SSDT HOOK实现进程保护和进程隐藏
  8. (转)淘淘商城系列——SSM框架整合之Dao层整合
  9. 【转】自动识别是手机端还是pc端只用一行代码就搞定
  10. Linux 控制终端转义和控制序列