Ps:流就是指一连串流动的字符,以先进先出的方式发送信息的通道。
Ps:文件输入——读、文件输出——写。
packagecom.imooc.file; importjava.io.File; importjava.io.IOException; publicclassFileDemo { publicstaticvoidmain(String[] args) { // 创建 File 对象 Filefile1=newFile("c:\\imooc\\io\\score.txt"); // 方法1Filefile2=newFile("c:\\imooc","io\\score.txt"); // 方法2Filefile3=newFile("c:\\imooc"); // 方法3Filefile4=newFile(file3,"c:\\imooc"); // 判断是文件 or 目录,返回 false 的两种情况: // 1、字面意思,非我族类 // 2、路径不存在 System.out.println("is 目录:"+file4.isDirectory()); System.out.println("is 文件:"+file4.isFile()); // 创建目录 Filefile5=newFile("c:\\imooc\\set\\HashSet"); Filefile6=newFile("c:\\imooc\\set\\HashSet\\asd.txt"); if(!file5.exists()) { // file5.mkdir(); // 失败,只能创建单级目录 file5.mkdirs(); // 可创建多级目录 file6.mkdirs(); // asd.txt 是目录不是文件 } // 创建文件 Filefile7=newFile("c:\\imooc\\set\\HashSet\\abc"); if(!file7.exists()) { try { file7.createNewFile(); // abc 是一个无后缀名的文件 } catch (IOExceptione) { e.printStackTrace(); } } } }
Ps:Windows目录“\\”(因为“\”转义字符,所以多加一个);Linux目录“/”即可。
Ps:主要关注文件、缓冲、对象 IO流。
// 文件内容:Hello,imooc!publicclassFileInputDemo1 { publicstaticvoidmain(String[] args) { // 创建一个FileInputStream对象 try { FileInputStreamfis=newFileInputStream("imooc.txt"); // 常规写 // intn=fis.read(); // while(n!=-1){ // System.out.print((char)n); // n=fis.read(); // } // 简写 intn=0; while((n=fis.read())!=-1){ System.out.print((char)n); } fis.close(); } catch (FileNotFoundExceptione) { e.printStackTrace(); } catch(IOExceptione){ e.printStackTrace(); } } }
publicclassFileInputDemo2 { publicstaticvoidmain(String[] args) { // 创建一个FileInputStream对象 try { FileInputStreamfis = newFileInputStream("imooc.txt"); byte[] b=newbyte[100]; fis.read(b,0,5); // 从下标为0位置开始存储,一共存储5个字符 System.out.println(newString(b)); // Hellofis.close(); } catch (FileNotFoundExceptione) { e.printStackTrace(); } catch (IOExceptione) { e.printStackTrace(); } } }
publicclassFileOutputDemo { publicstaticvoidmain(String[] args) { FileOutputStreamfos; FileInputStreamfis; try { fos = newFileOutputStream("imooc.txt",true); // 第二参数,是否追加,默认为重新覆盖 fis = newFileInputStream("imooc.txt"); // 可能有编码问题,但是不影响写入读取的最终结果 fos.write(50); // 每次写一个字节 fos.write('a'); System.out.println(fis.read()); // 每次读一个字节 System.out.println((char)fis.read()); fos.close(); fis.close(); } catch (FileNotFoundExceptione) { e.printStackTrace(); } catch (IOExceptione) { e.printStackTrace(); } } }
/* 图片Copy */publicclassFileOutputDemo1 { publicstaticvoidmain(String[] args) { // 文件拷贝 try { FileInputStreamfis=newFileInputStream("happy.gif"); FileOutputStreamfos=newFileOutputStream("happycopy.gif"); intn=0; byte[] b=newbyte[1024]; while((n=fis.read(b))!=-1){ // fos.write(b); // 之前都会填充满,最后一次会在实际大小中增加了 (1024 - 最后一次实际内容) 的大小,所以可能会比原先大个 1KBfos.write(b,0,n); // 第三个参数 n 代表:实际大小,这样就不会造成空间的浪费 } fis.close(); fos.close(); } catch (FileNotFoundExceptione) { e.printStackTrace(); } catch(IOExceptione){ e.printStackTrace(); } } }
importjava.io.BufferedInputStream; importjava.io.BufferedOutputStream; importjava.io.FileInputStream; importjava.io.FileNotFoundException; importjava.io.FileOutputStream; importjava.io.IOException; publicclassBufferedDemo { publicstaticvoidmain(String[] args) { try { FileOutputStreamfos=newFileOutputStream("imooc.txt"); BufferedOutputStreambos=newBufferedOutputStream(fos); FileInputStreamfis=newFileInputStream("imooc.txt"); BufferedInputStreambis=newBufferedInputStream(fis); // 测试使用BufferedIO效率更高 longstartTime=System.currentTimeMillis(); bos.write(50); bos.write('a'); bos.flush(); System.out.println(bis.read()); System.out.println((char)bis.read()); longendTime=System.currentTimeMillis(); System.out.println(endTime-startTime); fos.close(); bos.close(); fis.close(); bis.close(); } catch (FileNotFoundExceptione) { e.printStackTrace(); } catch (IOExceptione) { e.printStackTrace(); } } }
Ps1:如果是边读边写,就会很慢,也伤硬盘。缓冲区就是内存里的一块区域,把数据先存内存里,然后一次性写入,类似数据库的批量操作,这样效率比较高。
Ps2:flush() 和 close() 都可以对写入缓冲(内存)的数据写进硬盘,如果没这两个方法,或者也没达到缓冲区满(满了不用 flush 也会自动进硬盘的),那么硬盘上的数据还是没有的。
缓冲字符流
/* 字节字符转换流 + 缓冲区 */importjava.io.BufferedReader; importjava.io.BufferedWriter; importjava.io.FileInputStream; importjava.io.FileNotFoundException; importjava.io.FileOutputStream; importjava.io.IOException; importjava.io.InputStreamReader; importjava.io.OutputStreamWriter; publicclassReaderDemo { publicstaticvoidmain(String[] args) { try { FileInputStreamfis=newFileInputStream("imooc.txt"); InputStreamReaderisr=newInputStreamReader(fis,"GBK"); // 不写编码方式,默认为IDE的编码;用什么编码方式读取,下面就对应用什么编码方式写入 BufferedReaderbr=newBufferedReader(isr); FileOutputStreamfos=newFileOutputStream("imooc1.txt"); OutputStreamWriterosw=newOutputStreamWriter(fos,"GBK"); BufferedWriterbw=newBufferedWriter(osw); intn=0; char[] cbuf=newchar[10]; // while((n=isr.read())!=-1){ // 这里的 n 代表实际内容 // System.out.print((char)n); // } // while((n=isr.read(cbuf))!=-1){ // 这里的 n 代表实际内容大小 // Strings=newString(cbuf,0,n); // System.out.print(s); // } while((n=br.read(cbuf))!=-1){ //Strings=newString(cbuf,0,n); bw.write(cbuf, 0, n); } bw.flush(); // 注意关闭顺序 br.close(); bw.close(); isr.close(); osw.close(); fis.close(); fos.close(); } catch (FileNotFoundExceptione) { e.printStackTrace(); } catch (IOExceptione) { e.printStackTrace(); } } }
Ps:FileInputStream + InputStreamReader == FileReader,依此类推。
- 序列化:把Java对象转换为字节序列的过程。
- 反序列化:把字节序列恢复为Java对象的过程。
importjava.io.Serializable; publicclassGoodsimplementsSerializable{ privateStringgoodsId; privateStringgoodsName; privatedoubleprice; publicGoods(StringgoodsId,StringgoodsName,doubleprice){ this.goodsId=goodsId; this.goodsName=goodsName; this.price=price; } // getters/setters... @OverridepublicStringtoString() { return"商品信息 [编号:" + goodsId + ", 名称:" + goodsName + ", 价格:" + price + "]"; } }
importjava.io.FileInputStream; importjava.io.FileNotFoundException; importjava.io.FileOutputStream; importjava.io.IOException; importjava.io.ObjectInputStream; importjava.io.ObjectOutputStream; publicclassGoodsTest { publicstaticvoidmain(String[] args) { // 定义Goods类的对象 Goodsgoods1 = newGoods("gd001", "电脑", 3000); try { FileOutputStreamfos = newFileOutputStream("imooc.txt"); ObjectOutputStreamoos = newObjectOutputStream(fos); FileInputStreamfis = newFileInputStream("imooc.txt"); ObjectInputStreamois = newObjectInputStream(fis); // 将Goods对象信息写入文件,如果文件里面是乱码不要紧,只要读出来显示正常即可 oos.writeObject(goods1); oos.writeBoolean(true); oos.flush(); // 读对象信息 // 注意:读顺序和写顺序符合先进先出规则,如果把readBoolean方法放首先会RE,因为第一个是对象而不是booleanGoodsgoods = (Goods) ois.readObject(); System.out.println(goods); System.out.println(ois.readBoolean()); fos.close(); oos.close(); fis.close(); ois.close(); } catch (ClassNotFoundExceptione) { e.printStackTrace(); } catch (FileNotFoundExceptione) { e.printStackTrace(); } catch (IOExceptione) { e.printStackTrace(); } } }
- 序列化时,只对对象的状态进行保存,而不管对象的方法;
- 当一个父类实现序列化,子类自动实现序列化,不需要显式实现Serializable接口;
- 当一个对象的实例变量引用其他对象,序列化该对象时也把引用对象进行序列化;
- 并非所有的对象都可以序列化,至于为什么不可以,有很多原因。