一、File
File 类(磁盘操作)可以用于表示文件和目录的信息,但是它不表示文件的内容。递归地列出一个目录下所有文件:
public static void listAllFiles(File dir) {if (dir == null || !dir.exists()) {return;}if (dir.isFile()) {System.out.println(dir.getName());return;}for (File file : dir.listFiles()) {listAllFiles(file);}
}
二、IO 概述
什么是 IO: 磁盘与内存之间的数据交互,可以看做是一种数据的流动,按照流动方向,以内存为基准,分为输入 input 和输出 output,即流向内存的是输入流,流出内存的是输出流。Java IO操作主要是指使用 java.io包下的内容,进行输入、输出操作。输入也叫读取数据,输出也叫写出数据。
IO 的分类: 根据数据的流向分为:输入流和输出流。根据数据的类型分为:字节流(以字节为单位)和字符流(以字符为单位)。
输入流 | 输出流 | |
---|---|---|
字节流 | InputStream | OutputStream |
字符流 | Reader | Writer |
三、字节流
一切文件数据(文本、图片、视频)在存储时,都是以二进制数字的形式保存,传输时也是如此。所以,字节流可以传输任意文件数据,在操作流的时候,我们要时刻明确,无论使用什么样的流对象,底层传输始终为二进制数据。
字节输入流InputStream
:java.io.InputStream
抽象类是表示字节输入流的所有类的超类,可以读取字节信息到内存。它定义了字节输入流的基本共性。
public abstract int read() | 从输入流读取数据的下一个字节 |
public int read(byte b[] b) | 从输入流中读取一些字节数,并将它们存储到字节数组b中 |
public int read(byte b[] b, int off, int len) | off 写入数据的位置,数组中的起始偏移量。从输入流读取多达len字节的数据到字节数 |
public void close() | 关闭此输入流并释放与此流相关联的任何资源 |
// 读取数据
public abstract int read()// 将读取到的数据放在 byte 数组中,该方法实际上是根据下面的方法实现的,off 为 0,len 为数组的长度
public int read(byte b[])// 从第 off 位置读取 len 长度字节的数据放到 byte 数组中,流是以 -1 来判断是否读取结束的
public int read(byte b[], int off, int len)// 跳过指定个数的字节不读取,想想看电影跳过片头片尾
public long skip(long n)// 返回可读的字节数量
public int available()// 读取完,关闭流,释放资源
public void close()// 标记读取位置,下次还可以从这里开始读取,使用前要看当前流是否支持,可以使用 markSupport() 方法判断
public synchronized void mark(int readlimit)// 重置读取位置为上次 mark 标记的位置
public synchronized void reset()// 判断当前流是否支持标记流,和上面两个方法配套使用
public boolean markSupported()
字节输出流OutputStream
:java.io.OutputStream
抽象类是表示字节输出流的所有类的超类,它定义了字节流的基本共性功能方法。
public abstract void write(int b) | 将指定的字节写入此输出流 |
public void write(byte b[]) | 将 b.length字节从指定的字节数组写入此输出流 |
public void write(byte b[], int off, int len) | 从指定的字节数组写入 len字节,从偏移量 off开始输出到此输出流 |
public void flush() | 刷新此输出流并强制任何缓冲的输出字节被写出 |
public void close() | 关闭此输出流并释放此流相关联的任务系统资源 |
//写入一个字节,可以看到这里的参数是一个 int 类型,对应上面的读方法,int 类型的 32 位,只有低 8 位才写入,高 24 位将舍弃。
public abstract void write(int b)// 将数组中的所有字节写入,和上面对应的 read() 方法类似,实际调用的也是下面的方法。
public void write(byte b[])// 将 byte 数组从 off 位置开始,len 长度的字节写入
public void write(byte b[], int off, int len)// 强制刷新,将缓冲中的数据写入
public void flush()// 关闭输出流,流被关闭后就不能再输出数据了
public void close()
四、文件流
FileInputStream
把硬盘中的数据读取到内存中
/*** java.io.InputStream: 字节输入流。 此抽象类是表示字节输入流的所有类的超类。* java.io.FileInputStream extends InputStream: 文件输入流* 作用:把硬盘文件中的数据,读取到内存中** 构造方法:public FileInputStream(String name) 文件路径* public FileInputStream(File file) 文件* 构造方法的作用:* 1、创建一个 FileInputStream 对象;* 2、会把 FileInputStream对象指向构造方法中创建的文件* java程序--JVM--OS读取方法--读取文件* 1个中文 GBK占两个字节 UTF-8占3个字节*/
public class MyInputStream {public static void main(String[] args) {FileInputStream fis = null;/*** JDK7 新特性【了解】 在 try 后边可以增加一个() ,在括号中定义流对象,那么这个流对象的作用域就在 try中有效,* try 中的代码执行完毕,会自动把流对象释放,不用写 finally。多个流用 ;号隔开* 例如: try(FileInputStream fis = new FileInputStream("test.txt");* FileOutputStream fos = new FileOutputStream("test.txt");){}** JDK9 新特性【了解】* 在 try前边定义流对象* 在 try 括号中引入流对象名称 ,不用写 finally* 例如:* FileInputStream fis = new FileInputStream("test.txt");* FileOutputStream fos = new FileOutputStream("test.txt");* try(fis ; fos){}*/try {//1、创建 FileInputStream对象,参数为绑定的数据源fis = new FileInputStream("D:/test.txt");//2、读取文件 read ,当读到结尾的时候返回 -1 , 注意需要用一个变量来接收 read返回值,不要读两次/*int len = 0;while((len = fis.read())!= -1){System.out.println((char)len);};*//*** 一次读取多个字节的方法,byte[] 起到缓冲作用,存储每次读取到的字节个数。数组的长度一般定义为1024(1kb)或整数倍*/byte[] bytes = new byte[2];int lens = fis.read(bytes);System.out.println(lens);//2 每次读取字节的个数System.out.println(Arrays.toString(bytes)); //[97, 98]System.out.println(new String(bytes));//ab} catch (IOException e) {e.printStackTrace();}finally {if(fis != null){//3、关闭流try {fis.close();} catch (IOException e) {e.printStackTrace();}}}}
}
FileOutputStream
把内存中的数据写入到硬盘的文件中
/*** FileOutputStream: 文件输出流,将内存数据写入到文件中* 构造方法: FileOutputStream(String name) 传入一个文件的路径* FileOutputStream(File file) 传入一个文件* FileOutputStream(String name,boolean append) 当 append为true 的时候,就不会覆盖文件,继续在文件的末尾追加写数据* 构造方法的作用:* 1、创建一个 FileOutputStream 对象* 2、会根据构造方法中传递的文件/文件路径,创建一个空的文件* 3、会把 FileOutputStream 对象指向创建好的文件* 写数据的原理* java程序---JVM--- OS(操作系统)---将数据写入文件中* 写换行:换行符* windows \r\n* linux /n* max /r*/
public class MyOutputStream {public static void main(String[] args) {FileOutputStream fos = null;{try {//1、创建一个 FileOutputStream对象fos = new FileOutputStream("D:/test.txt");/*** 2、调用 FileOutputStream对象的 writer方法,将数据写入文件* 会把十进制的整数,转为二进制存入硬盘.任意的记事本,打开时都会查询编码表,把字节转为字符表示。查询 ASKII码* 所以 test.txt中存的是 d*/fos.write(100);//写出多个字节byte[] b = "你好".getBytes();fos.write(b);} catch (IOException e) {e.printStackTrace();}finally {if(fos != null){//3、释放资源,流使用会占用一定的内存,因此需要释放try {fos.close();} catch (IOException e) {e.printStackTrace();}}}}}
}
五、字符流
当使用字节流读取文本时,可能会有一个问题。就是遇到中文字符时,可能不会显示完整的字符,因为一个中文字符可能占用多个字节存储。所以 Java提供一些字符流,以字符为单位读写数据,专门用于处理文本文件。
字符输入流【Reader】:java.io.Reader 抽象类是读取字符流所有类的超类,可以读取字符信息到内存中。它定义了字符输入流的基本共性功能方法:
public int read() | 从输入流读取一个字符 |
public int read(char cbuf[]) | 从输入流中读取一些字符,并将它们存储到字符数组 cbuf中 |
abstract public void close() | 关闭此流并释放与此流相关联的任何系统资源 |
FileReader 文件字符输入流,继承 InputStreamReader 。
/*** 字符输入流 Reader 输入流的最顶层父类* FileReader 构造方法:* FileReader(String fileName)* FileReader(File file)* 作用:* 1、创建一个 FileReader 对象* 2、会把 FileReader 对象指向要读取的文件***/
public class MyReader {public static void main(String[] args) {FileReader fr = null;try {//1、创建 FileReader对象,参数中绑定读取的资源fr = new FileReader("test.txt");//2、使用 read读取当个字符并返回int len = 0;/*while((len = fr.read()) != -1){System.out.print((char)len); //你好!}*///读取多个字符char[] cs = new char[1024];while((len = fr.read(cs)) != -1){System.out.printf(new String(cs,0,len));}} catch (IOException e) {e.printStackTrace();}finally {if(fr != null){//3、释放资源try {fr.close();} catch (IOException e) {e.printStackTrace();}}}}
}
字符输出流【Writer】:java.io.Writer 抽象类是所有输出字符流的最顶层父类。它定义了字符输入流的基本共性功能方法:
public void write(int c) | 写入单个字符 |
public void write(char cbuf[]) | 写入字符数组 |
abstract public void write(char cbuf[], int off, int len) | 写入字符数组的某一部分 |
public void write(String str) | 写入字符串 |
public void write(String str, int off, int len) | 写入字符串的某一部分 |
abstract public void flush() | 刷新流的缓冲 |
abstract public void close() | 关闭此流,但先要刷新它 |
六、Properties 集合
/*** java.util.Properties 集合 extends Hashtable<k,v>* Properties 类表示一个持久的属性集。Properties 可保存在流中或从流中加载。* Properties 集合是一个唯一和 IO流相结合的集合,* 可以使用集合中的方法 store() ,把集合中的临时数据,持久化到硬盘中* load(), 把硬盘中的数据读取到集合中* Properties 集合是一个双列集合,key 和 value 都是字符串**/
public class MyProperties {public static void main(String[] args) {// 使用 Properties 集合存储数据,遍历取出数据//1、创建 Properties 对象Properties prop = new Properties();//2、添加数据prop.setProperty("you","小三");prop.setProperty("mine","正房");prop.setProperty("she","小四");/*** 3、将数据写入到磁盘中保存* store(OutputStream out, String comments)* store(Writer writer, String comments) 可以写中文*/FileReader fr = null;FileWriter fw = null;try {fw = new FileWriter("test.txt");fr = new FileReader("test.txt");/*** 文件中的内容如下:* #save data* #Sun Nov 01 16:26:13 CST 2020* you=小三* she=小四* mine=正房*/prop.store(fw, "save data"); //第二个参数是注释/*** load 将文件中的数据读取:* 存储键值对的文件中,键与值默认的连接符号可以是 =,空格(其它符号)* # 表示注释* 键和值默认都是字符串*/Properties propRead = new Properties();propRead.load(fr);//将在数据到 PropertiesSet<String> strings = propRead.stringPropertyNames();//获取 key列表/*** 输入为:* you = 小三* she = 小四* mine = 正房*/for(String key: strings){System.out.println(key + " = " + propRead.get(key));}} catch (IOException e) {e.printStackTrace();}finally {if(fw != null && fr != null){try {fw.close();fr.close();} catch (IOException e) {e.printStackTrace();}}}}
}
七、缓冲流
缓冲流也叫高效流,是对 4个基本的 Filexxx流的增强,所以也是4个流,按照数据类型分为:
【1】字节缓冲流:BufferedInputStream,BufferedOutputStream
【2】字符缓冲流:BufferedReader,BufferedWriter
缓冲流的基本原理,是在创建流对象时,会创建一个内置的默认大小的缓冲数组,通过缓冲区读写,减少系统 IO次数,从而提高效率。
字节缓冲输出流:BufferedOutputStream
/*** 继承 OutputStream 提高写入效率* 构造方法:* BufferedOutputStream(OutputStream out) 创建一个新的缓冲区输出流,以将数据写入指定的底层输出流,默认缓冲区大小= 8192* BufferedOutputStream(OutputStream out, int size) 创建一个新的缓冲区输出流,以将具体指定缓冲区大小的数据写入底层输出流*/
public class MyBufferedOutputStream {public static void main(String[] args) {FileOutputStream fos = null;BufferedOutputStream bos = null;try {//1、创建字节输出流 FileOutputStreamfos = new FileOutputStream("test.txt");//2、创建缓冲流 BufferedOutputStreambos = new BufferedOutputStream(fos);//3、使用 writer 方法写入内部缓冲区中bos.write("使用内部缓冲区".getBytes());//4、flush 将内部缓冲区的数据刷新到文件中bos.flush();} catch (IOException e) {e.printStackTrace();}finally {if(bos != null){//5、close 关闭流,会先执行 flush后关闭try {bos.close(); //会自动关闭 fos.close();} catch (IOException e) {e.printStackTrace();}}}}
}
字节缓冲输入流:BufferedInputStream
/*** 继承 InputStream 称为 字节缓冲输入流* 构造方法:* BufferedInputStream(InputStream in)* public BufferedInputStream(InputStream in, int size)*/
public class MyBufferedInputStream {public static void main(String[] args) {//1、创建 FileInputStream对象FileInputStream fis = null;BufferedInputStream bis = null;try {fis = new FileInputStream("test.txt");//2、创建 BufferedInputStream 对象,提高 FileInputStream对象读取的效率bis = new BufferedInputStream(fis);//3、使用 read方法读取数据int len = 0;byte[] bytes = new byte[1024];while((len = bis.read(bytes)) != -1){System.out.println(new String(bytes,0,len));}} catch (IOException e) {e.printStackTrace();}finally {//4、关闭流if(bis != null){try {bis.close(); //自动关闭 fis} catch (IOException e) {e.printStackTrace();}}}}
}
BufferedWriter :新方法 newLine() 表示换行,替换了 “\r\n" BufferedReader:新方法 readLine() 读取一个文本行,读取一行数据。如果到达流末尾,则返回 null。
八、转换流
字符编码 Character Encoding:就是一套自然语言的字符与二进制数之间的对应规则;
字符集 Charset:也叫编码表。是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等。常见字符集有 ASCII、GBK、Unicode字符集。
在 IDEA中,使用 FileReader读取项目中的文本文件,由于 IDEA 默认使用 UTF-8编码,所有没有任何问题。但是当读取 Windows系统中创建的文本文件时,由于 Windows系统的默认是 GBK编码,就会出现乱码。
转化流 InputStreamReader:是 Reader的子类,是从字节流到字符流的桥梁,他读取字节,并使用指定的字符集将其解码为字符,它的字符集可以由名称指定,也可以接收平台默认的字符集。
/*** InputStreamReader:是 Reader的子类,是从字节流到字符流的桥梁,他读取字节,并使用指定的字符集将其解码为字符,它的字符集可以由名称指定,也可以接收平台默认的字符集。* 构造方法:* InputStreamReader(InputStream in)* InputStreamReader(InputStream in, String charsetName)*/
public class MyInputStreamReader {public static void main(String[] args) {//1、创建 InputStreamReader对象,构造方法中传入字节输入流和指定的编码表名称 指定的编码要与文件的编码相同,否则会发生乱码;InputStreamReader isr = null;try {isr = new InputStreamReader(new FileInputStream("text.txt"), "GBK");//2、使用 read 方法读取文件int len = 0;while((len = isr.read()) != -1){System.out.println((char)len);}} catch (IOException e) {e.printStackTrace();}finally {if(isr != null){//3、释放资源try {isr.close();} catch (IOException e) {e.printStackTrace();}}}}
}
OutputStreamWriter:是 Writer的子类,是从字符流到字节流的桥梁,他写入数据,并使用指定的字符集将其编码为字符,它的字符集可以由名称指定,也可以接收平台默认的字符集。
/*** java.io.OutputStreamWriter 继承了 Writer* OutputStreamWriter 是字符流通向字节流的桥梁,可使用指定的 charset 将要写入流中的字符编码成字节。* 构造方法:* OutputStreamWriter(OutputStream out) 使用默认的字符编码* OutputStreamWriter(OutputStream out, Charset cs) 可以指定字符编码,不区分大小写**/
class MyOutputStreamWriter {public static void main(String[] args) {//1、创建一个 OutputStreamWriter 对象OutputStreamWriter osw = null;try {osw = new OutputStreamWriter(new FileOutputStream("text.txt"), "GBK");//2、使用 writer 写出数据osw.write("你好");//3、flush 刷新数据osw.flush();} catch (UnsupportedEncodingException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}finally {//4、关闭流if(osw != null){try {osw.close();} catch (IOException e) {e.printStackTrace();}}}}
}
九、序列化流
把对象以流的方式写入到文件中保存,对象中不仅仅是字符,也有字节。就有了 ObjectOutputStream[对象的序列化 writeObject§] 和 ObjectInputStream[对象的反序列化 readObject()]
ObjectOutputStream 对象序列化流:
/*** java.io.ObjectOutputStream extends OutputStream* 构造器:* ObjectOutputStream(OutputStream out) 创建写入指定 OutputStream 的 ObjectOutputStream* 特有的成员方法:writeObject*/
public class MyObjectOutputStream {public static void main(String[] args) {//1、创建 ObjectOutputStream 对象,构造方法中传递字节输出流ObjectOutputStream oos = null;try {oos = new ObjectOutputStream(new FileOutputStream("text.txt"));//2、通过 writeObject 把对象写入到文件中oos.writeObject(new Person("张三",1));//通过二进制序列化流存储的,无法打开查看。} catch (IOException e) {e.printStackTrace();}finally {//3、关闭流try {oos.close();} catch (IOException e) {e.printStackTrace();}}}
}
ObjectInputStream 对象反序列化流:
/*** java.io.ObjectInputStream 继承了 InputStream 将对象从文件中读取使用* 构造方法:* ObjectInputStream(InputStream in) 创建从指定 InputStream 读取的 ObjectInputStream* 特有的方法:* readObject() 返回一个 object对象** transient 关键字:* static 关键字:静态关键字,被 static修饰的成员变量不能被序列化的,序列化的都是对象的默认值。transient 与 static的作用是一样的,但却没有 static的含义* Person对象需要实现 Serializable 并且要定义序列化ID* 集合类的存储和对象相同*/
public class MyObjectInputStream {public static void main(String[] args) {//1、创建 ObjectInputStream对象,构造方法中传递字节输入流ObjectInputStream ois = null;try {ois = new ObjectInputStream(new FileInputStream("text.txt"));//2、使用 readObject 方法读取对象Person o = (Person)ois.readObject();//4、对象使用} catch (Exception e) {e.printStackTrace();}finally {if(ois != null){//3、释放资源try {ois.close();} catch (IOException e) {e.printStackTrace();}}}}
}
十、打印流
PrintStream:我们常用的 System.out.println 就是打印流。
/*** 可以打印各种数据值表示形式,不会抛出 IOException,有特有的方法 print println* 构造方法:* PrintStream(File file) 输出的是一个文件* PrintStream(OutputStream out) 输出的是一个字节输出流* PrintStream(String fileName) 输出的是一个文件路径* 继承了 OutputStream 拥有该成员中的方法* 注意事项:如果使用继承自父类中的 write方法写数据,那么会查询编码表。如果使用 print 就会原样使用。* 可以改变输出语句的目的地:System.setOut() 目的地通过参数传递*/
public class MyPrintStream {public static void main(String[] args) throws FileNotFoundException {//创建打印流,绑定要输出的目的地PrintStream ps = new PrintStream("text.txt");//使用 write 和 print写数据ps.write(97);//文件中为 aps.print(97);//文件中为 97//设置输出流打印的目的地System.setOut(ps);System.out.println("此打印信息会在 text.txt文件中显示");//释放资源ps.close();}
}
十一、网络通信
【1】InetAddress:用于表示网络上的硬件资源,即 IP地址。没有公有的构造函数,只能通过静态方法来创建实例。
InetAddress.getByName(String host);
InetAddress.getByAddress(byte[] address);
【2】URL:统一资源定位符。可以直接从 URL中读取字节流数据。
public static void main(String[] args) throws IOException {URL url = new URL("http://www.baidu.com");/* 字节流 */InputStream is = url.openStream();/* 字符流 */InputStreamReader isr = new InputStreamReader(is, "utf-8");/* 提供缓存功能 */BufferedReader br = new BufferedReader(isr);String line;while ((line = br.readLine()) != null) {System.out.println(line);}br.close();
}
【3】Sockets:使用 TCP协议实现网络通信;ServerSocket:服务器端类,Socket:客户端类。服务器和客户端通过 InputStream 和 OutputStream 进行输入输出。
【4】Datagram:使用 UDP协议实现网络通信。DatagramSocket:通信类。DatagramPacket:数据包类。
思考: Java 字节读取流的 read方法返回 int的原因?
读取二进制数据按字节读取,每次读一个字节(byte)。read()的底层是由 C++实现的,返回的是 unsigned byte,取值范围为[0~255],在 java中 byte只能表示 [-128,127]的无符号数,所以只能用 int类型接收,Java接收得到的就是 int[0~255]。