Java IO留存查看
IO也可以写作为 "
i/O
" ,也可以理解为In
和Out
,即输入与输出。所以,IO体系的基本功能就是: 读和写。IO流作用:读写设备上的东西,硬盘文件、内存、键盘、网络...
根据数据的走向,可分为:输入流,输出流。
根据处理的数据类型,可分为:字节流、字符流。
字节流和字符流的区别
- 字节流可以处理所有类型的数据,如MP3、图片、文字、视频等。在读取时。读到一个字节就返回一个字节。
在Java 中对应的类都以“Stream”结尾。
- 字符流仅能够处理纯文本数据,如txt文本等。在读取时,读到一个或者多个字节,先查找指定的编码表,然后将查到的字符返回。
在Java中对应的类都以”Reader“ 或 "Writer" 结尾。
其实Java流我们只要了解节点流(文件FileXXX,内存ByteArrayXXXX),是主要的,其余都是包装包裹着(为了更方便操作节点流)这个流进行的!
文件 FileXXX
内存 ByteArrayXXX
缓冲 BufferedXXX
字节流 xxxStream结尾的类
字符流 xxxWriter xxxReader结尾的类
字节字符流转换流 OutputStreamWriter(输出必定写) InputStreamReader(输入必定读)
一.字节输入输出流
字符流和字节流的主要区别:
a.字节流读取的时候,读到一个字节就返回一个字节; 字符流使用了字节流读到一个或多个字节(中文对应的字节数是两个,在UTF-8码表中是3个字节)时。先去查指定的编码表,将查到的字符返回。
b.字节流可以处理所有类型数据,如:图片,MP3,AVI视频文件,而字符流只能处理字符数据。只要是处理纯文本数据,就要优先考虑使用字符流,除此之外都用字节流。
IO流主要可以分为节点流和处理流两大类。
1.1、节点流类型
该类型可以从或者向一个特定的地点或者节点读写数据。主要类型如下:
1.2、处理流类型
1.2.1缓冲流(BufferedInPutStream/BufferedOutPutStream和BufferedWriter/BufferedReader)他可以提高对流的操作效率。
写入缓冲区对象:
BufferedWriter bufw=new BufferedWriter(new FileWriter("buf.txt"));
读取缓冲区对象:
BufferedReader bufr=new BufferedReader(new FileReader("buf.txt"));
该类型的流有一个特有的方法:readLine();一次读一行,到行标记时,将行标记之前的字符数据作为字符串返回,当读到末尾时,返回null,其原理还是与缓冲区关联的流对象的read方法,只不过每一次读取到一个字符,先不进行具体操作,先进行临时储存,当读取到回车标记时,将临时容器中储存的数据一次性返回。
1.2.2转换流(InputStreamReader/OutputStreamWriter)
InputStreamReader 从字节流到字符流的桥梁
OutputStreamWriter 从字符流到字节流的桥梁
该类型时字节流和字符流之间的桥梁,该流对象中可以对读取到的字节数据进行指定编码的编码转换。
构造函数主要有:
InputStreamReader(InputStream); //通过构造函数初始化,使用的是本系统默认的编码表GBK。
InputStreamWriter(InputStream,String charSet); //通过该构造函数初始化,可以指定编码表。
OutputStreamWriter(OutputStream); //通过该构造函数初始化,使用的是本系统默认的编码表GBK。
OutputStreamwriter(OutputStream,String charSet); //通过该构造函数初始化,可以指定编码表。
注意:在使用FileReader操作文本数据时,该对象使用的时默认的编码表,即
FileReader fr=new FileReader(“a.txt”); 与 InputStreamReader isr=new InputStreamReader(new FileInputStream("a.txt")); 的意义相同。如果要使用指定表编码表时,必须使用转换流,即如果a.txt中的文件中的字符数据是通过utf-8的形式编码,那么在读取时,就必须指定编码表,那么转换流时必须的。即:
InputStreamReader isr=new InputStreamReader(new FileInputStream("a.txt"),utf-8);
1.2.3数据流(DataInputStream/DataOutputStream)
该数据流可以方便地对一些基本类型数据进行直接的存储和读取,不需要再进一步进行转换,通常只要操作基本数据类型的数据,就需要通过DataStream进行包装。
构造方法:
DataInputStreamReader(InputStream);
DataInputStreamWriter(OutputStream);
注意:在使用数据流读/存数据的时候,需要有一定的顺序,即某个类型的数据先写入就必须先读出,服从先进先出的原则。
二.字符输入输出流
2.1输入流Reader
2.2输出流Writer
2.3字节字符转换:
2.3.1:字节输入流转换为字符输入流:
InputStreamReader是字节流向字符流的桥梁,它使用指定的charset读取字节并将其解码为字符,它使用的字符集可以由名称指定或显示给定。根据InputStream的实例创建InputStreamReader的方法有4种:
InputStreamReader(InputStream in)//根据默认字符集创建
InputStreamReader(InputStream in,Charset cs)//使用给定字符集创建
InputStreamReader(InputStream in,CharsetDecoder dec)//使用给定字符集解码器创建
InputStreamReader(InputStream in,String charsetName)//使用指定字符集创建
2.3.2:字节输出流转换为字符输出流
OutputStreamWriter是字符流通向字节流的桥梁,它使用指定的charset将要写入流中的字符编码成字节,它使用的字符集可以由名称指定或显示给定,否则将接受默认的字符集:
根据根据InputStream的实例创建OutputStreamWriter的方法有4种:
OutputStreamWriter(outputstream out)//根据默认的字符集创建
OutputStreamWriter(outputstream out,charset cs)//使用给定的字符集创建
OutputStreamWriter(outputstream out,charsetDecoder dec)//使用组定字符集创建
OutputStreamWriter(outputstream out,String charsetName)//使用指定字符集创建
三、打印流(PrintStream/PrintWriter)
PrintStream是一个字节打印流,System.out对应的类型就是PrintStream,它的构造函数可以接受三种数据类型的值:1.字符串路径。2.File对象 3.OutputStream
PrintStream是一个字符打印流,它的构造函数可以接受四种类型的值:
1.字符串路径。 2.File对象 3.OutputStream
4.Writer 对于1、2类型的数据,可以指定编码表,也就是字符集,对于3、4类型的数据,可以指定自动刷新,当该自动刷新为True时,只有3个方法可以用:println,printf,format。
四、对象流(ObjectInputStream/ObjectOutputStream)
该类型的流可以把类作为一个整体进行存取,主要方法有:
Object readObject();该方法抛出异常:ClassNotFountException。
void writeObject(Object):被写入的对象必须实现一个接口:Serializable,否则就会抛出:NotSerializableException
---------------------
部分原文摘自:https://blog.csdn.net/zhengchao1991/article/details/53033137
自己留存代码
A.复制文件
// 复制文件
// @srcFileName 源文件名
// @dstFileName 生成文件名
public static void copy(String srcFileName, String dstFileName) {
// 1.数据源
File src = new File(srcFileName);
File dst = new File(dstFileName);
FileInputStream fiStream = null;
FileOutputStream foStream = null;
try {
// 2.选择流
fiStream = new FileInputStream(src);
foStream = new FileOutputStream(dst);
// 3.操作
int len = -1;
// 测试小缓冲
// byte[] flush = new byte[4];
byte[] flush = new byte[1024];
while ((len = fiStream.read(flush)) != -1) {
foStream.write(flush, 0, len);
}
foStream.flush();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
// 4.释放资源
if (null != foStream) {
try {
foStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (null != fiStream) {
try {
fiStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
B.统计文件夹的大小
public static void count(File src) {
if (null != src && src.exists()) {
if (src.isFile()) { // 文件大小
len += src.length();
} else { // 子孙级
for (File subFile : src.listFiles()) {
count(subFile);
}
}
}
}
C.打印数值(用递归)
// 打印beginNum-endNum以内数字
public static void printNumber(int beginNum, int endNum) {
if (beginNum > endNum) { // 结束递归
return;
}
System.out.println(beginNum);
printNumber(beginNum + 1, endNum); // 方法调用自己
}
D.打印文件夹内的所有子文件
// 打印文件夹内的所有子文件
public static void printFileName(File src, int deep) {
for (int i = 0; i < deep; i++) {
System.out.print("-");
}
System.out.println(src.getName());
if (null == src || !src.exists()) {
return;
} else if (src.isDirectory()) { // 目录
for (File subFile : src.listFiles()) {
printFileName(subFile, deep + 1);
}
}
}