Java IO 流中共涉及 40 多个类,但都是有下面这四个类派生出来:
输入过程:
public void testFileReader(){ FileReader fr = null; try{ File file = new File("hello.txt"); fr = new FileReader(file); char[] cbuf = new char[5]; int len; //read(char[] cbuf):返回每次读入cbuf数组中的字符的个数。如果达到文件末尾,返回-1 while((len = fr.read(cbuf)) != -1){ String str = new String(cbuf,0,len); System.out.print(str); } }catch(IOException e){ e.printStackTrace(); }finally{ if(fr != null){ try{ fr.close(); }catch (IOException e){ e.printStackTrance(); } } } }
输出过程:
public void testFileWriter(){ FileWriter fw = null; try{ File file = new File("hello1.txt"); fw = new FileWriter(file); fw.write("Hello Java\n"); fw.wrrite("从入门到放弃"); }catch(IOException e){ e.printStackTrance(); }finally{ if(fw != null){ try{ fw.close(); }catch(IOException e){ e.printStackTrace(); } } } }
对于文本文件(.txt,java等),使用字符流处理。
对于非文本文件(.jpg,.mp4等),使用字节流处理。
两个方法:
例子:实现对图片的复制操作。
public void testFileInputOutputStream(){ FileInputStream fis = null; FileOutputStream fos = null; try{ File srcFile = new File("1.jpg"); File destFile = new File("2.jpg"); fis = new FileInputStream(srcFile); fos = new FileOutputStream(destFile); byte[] buffer = new byte[5]; int len; while((len = fis.read(buffer)) != -1){ fos.write(buffer,0,len); } }catch(IOException e){ e.printStackTrace(); }finally{ if(fos != null){ try{ fos.close(); }catch(IOException e){ e.printStackTrace(); } } if(fis != null){ try{ fis.close(); }catch(IOException e){ e.printStackTrace(); } } } }
涉及到的类:
BufferedInputStream
BufferedOutputStream
BufferedReader
BufferedWriter
作用:内部提供一个缓冲区,默认 8k;它提升流的读取、写入的速度。
例子:使用 BufferInputStream 和 BufferOutputStream :处理非文本文件。
//实现文件复制的方法 public void copyFileWrithBuffered(String srcPath,String destPath){ BufferedInputStream bis = null; BufferedOutputStream bos = null; try{ FileInputStream fis = new FileInputStream(srcFile); FileOutputStream fos = new FileOutputStream(destFile); //造缓冲流 bis = new BufferedInputStream(fis); bos = new BufferedOutputStream(fos); byte[] buffer = new byte[1024]; int len; while((len = bis.read(buffer)) != -1){ bos.write(buffer,0,len); } }catch(IOException e){ e.printStackTrace(); }finally{ if(bos != null){ try{ bos.close(); }catch(IOException e){ e.printStackTrace(); } if(bis != null){ try{ bos.close(); }catch(IOException e){ e.printStackTrace(); } } }
例子:使用 BufferedReader 和 BufferedWriter:处理文本文件
public void testBufferedReaderBufferedWriter(){ BufferedReader br = null; BufferedWriter bw = null; try { //创建文件和相应的流 br = new BufferedReader(new FileReader(new File("dbcp.txt"))); bw = new BufferedWriter(new FileWriter(new File("dbcp1.txt"))); //读写操作 //方式一:使用char[]数组 // char[] cbuf = new char[1024]; // int len; // while((len = br.read(cbuf)) != -1){ // bw.write(cbuf,0,len); // // bw.flush(); // } //方式二:使用String String data; while((data = br.readLine()) != null){ //方法一: // bw.write(data + "\n");//data中不包含换行符 //方法二: bw.write(data);//data中不包含换行符 bw.newLine();//提供换行的操作 } } catch (IOException e) { e.printStackTrace(); } finally { //关闭资源 if(bw != null){ try { bw.close(); } catch (IOException e) { e.printStackTrace(); } } if(br != null){ try { br.close(); } catch (IOException e) { e.printStackTrace(); } } } }
转换流涉及到的类:输入字符流。
说明:编码决定了解码的方式
作用:提供字节流和字符流之间的转换:
例子:
public void test1() throws IOException { FileInputStream fis = new FileInputStream("dbcp.txt"); // InputStreamReader isr = new InputStreamReader(fis);//使用系统默认的字符集 //参数2指明了字符集,具体使用哪个字符集,取决于文件dbcp.txt保存时使用的字符集 InputStreamReader isr = new InputStreamReader(fis,"UTF-8");//使用系统默认的字符集 char[] cbuf = new char[20]; int len; while((len = isr.read(cbuf)) != -1){ String str = new String(cbuf,0,len); System.out.print(str); } isr.close(); } /* 此时处理异常的话,仍然应该使用try-catch-finally 综合使用InputStreamReader和OutputStreamWriter */ @Test public void test2() throws Exception { //1.造文件、造流 File file1 = new File("dbcp.txt"); File file2 = new File("dbcp_gbk.txt"); FileInputStream fis = new FileInputStream(file1); FileOutputStream fos = new FileOutputStream(file2); InputStreamReader isr = new InputStreamReader(fis,"utf-8"); OutputStreamWriter osw = new OutputStreamWriter(fos,"gbk"); //2.读写过程 char[] cbuf = new char[20]; int len; while((len = isr.read(cbuf)) != -1){ osw.write(cbuf,0,len); } //3.关闭资源 isr.close(); osw.close(); }
对象流:ObjectInputStream 和 ObjectOutputStream
序列化和反序列化:
① Java 序列化就是指把 Java 对象转换为字节序列的过程。
② Java反序列化就是指把字节序列恢复为Java对象的过程。
作用:
① ObjectOutputStream:内存中的对象—>存储中的文件、通过网络传输出去:序列化过程。
② ObjectInputStream:存储中的文件、通过网络接收过来 —>内存中的对象:反序列化过程。
对象序列化机制:对象序列化机制允许把内存中的Java对象转换成平台无关的二进制流,从而允许把这种二进制流持久地保存在磁盘上,或通过网络将这种二进制流传输到另一个网络节点。当其它程序获取了这种二进制流,就可以恢复成原来的Java对象。
序列化代码实现:
① 实现序列化的对象所属的类需要满足:实现了 Serializable 接口、当前类提供一个全局常量 serialVersionUID;
② 除了当前类需要实现 Serializable 接口外,还必须保证内部的属性也必须是可序列化的。默认情况下,基本数据类型是可序列化。
③ ObjectInputStream 和 ObjectOutputStream 不能序列化 static 和 transient 修饰的成员变量。
④ 代码:
@Test public void testObjectOutputStream(){ ObjectOutputStream oos = null; try { //1. oos = new ObjectOutputStream(new FileOutputStream("object.dat")); //2. oos.writeObject(new String("我爱北京天安门")); oos.flush();//刷新操作 oos.writeObject(new Person("王铭",23)); oos.flush(); oos.writeObject(new Person("张学良",23,1001,new Account(5000))); oos.flush(); } catch (IOException e) { e.printStackTrace(); } finally { if(oos != null){ //3. try { oos.close(); } catch (IOException e) { e.printStackTrace(); } } } }
@Test public void testObjectInputStream(){ ObjectInputStream ois = null; try { ois = new ObjectInputStream(new FileInputStream("object.dat")); Object obj = ois.readObject(); String str = (String) obj; Person p = (Person) ois.readObject(); Person p1 = (Person) ois.readObject(); System.out.println(str); System.out.println(p); System.out.println(p1); } catch (IOException e) { e.printStackTrace(); } catch (ClassNotFoundException e) { e.printStackTrace(); } finally { if(ois != null){ try { ois.close(); } catch (IOException e) { e.printStackTrace(); } } } }
1、随机存取文件流:RandomAccessFile
2、使用说明:
RandomAccessFile直接继承于java.lang.Object类,实现了DataInput和DataOutput接口
RandomAccessFile既可以作为一个输入流,又可以作为一个输出流
如果RandomAccessFile作为输出流时,写出到的文件如果不存在,则在执行过程中自动创建。
如果写出到的文件存在,则会对原文件内容进行覆盖。(默认情况下,从头覆盖)
可以通过相关的操作,实现RandomAccessFile“插入”数据的效果。seek(int pos)
经典代码:
@Test public void test1() { RandomAccessFile raf1 = null; RandomAccessFile raf2 = null; try { //1. raf1 = new RandomAccessFile(new File("爱情与友情.jpg"),"r"); raf2 = new RandomAccessFile(new File("爱情与友情1.jpg"),"rw"); //2. byte[] buffer = new byte[1024]; int len; while((len = raf1.read(buffer)) != -1){ raf2.write(buffer,0,len); } } catch (IOException e) { e.printStackTrace(); } finally { //3. if(raf1 != null){ try { raf1.close(); } catch (IOException e) { e.printStackTrace(); } } if(raf2 != null){ try { raf2.close(); } catch (IOException e) { e.printStackTrace(); } } } }
经典代码二:
/* 使用RandomAccessFile实现数据的插入效果 */ @Test public void test3() throws IOException { RandomAccessFile raf1 = new RandomAccessFile("hello.txt","rw"); raf1.seek(3);//将指针调到角标为3的位置 //保存指针3后面的所数据到StringBuilder中 StringBuilder builder = new StringBuilder((int) new File("hello.txt").length()); byte[] buffer = new byte[20]; int len; while((len = raf1.read(buffer)) != -1){ builder.append(new String(buffer,0,len)) ; } //调回指针,写入“xyz” raf1.seek(3); raf1.write("xyz".getBytes()); //将StringBuilder中的数据写入到文件中 raf1.write(builder.toString().getBytes()); raf1.close(); //思考:将StringBuilder替换为ByteArrayOutputStream }
1、NIO的使用说明:
Java NIO (New IO,Non-Blocking IO)是从Java 1.4版本开始引入的一套新的IO API,可以替代标准的Java IO AP。
NIO与原来的IO同样的作用和目的,但是使用的方式完全不同,NIO支持面向缓冲区的(IO是面向流的)、基于通道的IO操作。
NIO将以更加高效的方式进行文件的读写操作。
随着 JDK 7 的发布,Java对NIO进行了极大的扩展,增强了对文件处理和文件系统特性的支持,以至于我们称他们为 NIO.2。
2、Path的说明:
Path替换原有的File类。
3、如何实例化:
Paths类提供的静态get()方法用来获取Path对象:
static Path get(String first, String … more):用于将多个字符串串连成路径>static Path get(URl uri):返回指定uri对应的Path路径
4、常用方法:
String toString():返回调用Path 对象的字符串表示形式
boolean startsWith(String path):判断是否以path 路径开始
boolean endsWith(String path):判断是否以path 路径结束
boolean isAbsolute()∶判断是否是绝对路径
Path getParent():返回Path对象包含整个路径,不包含Path对象指定的文件路径
Path getRoot():返回调用Path 对象的根路径
Path getFileName():返回与调用Path 对象关联的文件名
int getNameCount()∶返回Path 根目录后面元素的数量
Path getName(int idx):返回指定索引位置idx的路径名称
Path toAbsolutePath():作为绝对路径返回调用Path 对象
Path resolve(Path p):合并两个路径,返回合并后的路径对应的Path对象
File toFile():将Path转化为File类的对象
5、Files工具类—jdk7提供—常用方法
Path copy(Path src,Path dest,CopyOption … how):文件的复制
Path createDirectory(Path path, FileAttribute<?> ... attr):创建一个目录Path createFile(Path path,FileAttribute<?> … arr):创建一个文件
void delete(Path path):删除一个文件/目录,如果不存在,执行报错
void deletelfExists(Path path): Path对应的文件/目录如果存在,执行删除
Path move(Path src,Path dest, CopyOption…how):将src移动到dest位置
long size(Path path):返回path 指定文件的大小
Files常用方法:用于判断
boolean exists(Path path,LinkOption … opts):判断文件是否存在
boolean isDirectory(Path path,LinkOption … opts):判断是否是目录
boolean isRegularFile(Path path, LinkOption … opts):判断是否是文件
boolean isHidden(Path path):判断是否是隐藏文件
boolean isReadable(Path path):判断文件是否可读
boolean isWritable(Path path):判断文件是否可写
boolean notExists(Path path, LinkOption … opts):判断文件是否不存在
Files常用方法:用于操作内容
SeekableByteChannel newByteChannel(Path path, OpenOption…how):获取与指定文件的连接,how指定打开方式。
DirectoryStreamnewDirectoryStream(Path path):打开path 指定的目录
InputStream newInputStream(Path path, OpenOption…how):获取InputStream对象
OutputStream newOutputStream(Path path, OpenOption…how):获取OutputStream对象
参考: https://www.bilibili.com/video/BV1Kb411W75N?from=search&seid=8802845884941874275&spm_id_from=333.337.0.0