IT评测·应用市场-qidao123.com
标题:
java IO流
[打印本页]
作者:
民工心事
时间:
2023-5-23 16:53
标题:
java IO流
Java IO流
什么是流?
概念:内存和存储设备之间传输数据的通道。
数据借助流传输。
流分类:
按照方向:输入流(将存储设备中的内容读入到内存中)和输出流(将内存中的内容写入到存储设备中)
按照单位:字节流(以字节为单位,可以读写所有数据)和字符流(以字符为单位,只能读取文本数据)
按照功能:节点流(具有实际传输数据的读写功能)和过滤流(在节点流的基础之上增强功能)
字节流
字节流的父类(抽象类):
InputStream:字节输入流:提供一些如read(),close()的方法
OutputStream:字节输出流:提供一些如write(),close()的方法
文件字节流
FileInputStream:文件字节输入流,继承字节输入流
FileOutputStream:文件字节输出流,继承字节输入流
文件字节输入流操作:
//创建FileInputStream并且指定文件路径
FileInputStream fis = new FileInputStream("d:\\aa.txt");
int data = 0;
//read返回的int,当等于-1表示读完了,如果是字符返回对应的asc码
//一次读取一个字节
while ((data=fis.read())!=-1){
System.out.println((char) data);
}
/*
a
b
c
复制代码
什么为一次读取一个字节的情况
当需要一次读取多个字节
//创建FileInputStream并且指定文件路径
FileInputStream fis = new FileInputStream("d:\\aa.txt");
//一次读取多个个字节
byte[] buf = new byte[3];
//本处read返回实际读取个数,读完返回-1,并且将读取到的存储在buf中
//因为数组大小一次读取三个
System.out.println(fis.read(buf));
System.out.println(new String(buf));
fis.close();
/*
3
abc
*/
//或者
FileInputStream fis = new FileInputStream("d:\\aa.txt");
byte[] buf = new byte[3];
int count=0;
while ((count=fis.read(buf))!=-1){
System.out.println(new String(buf,0,count));
}
fis.close();
/*
abc
def
g
复制代码
本处只是例子,实际操作时缓冲区数组不需要创建得太小,可以是1024以上
文件字节输出流操作:
//创建文件字节输出流,为true代表文件已经存在时追加,false每次都会覆盖(默认)
FileOutputStream fos=new FileOutputStream("d://aa.txt",false);
fos.write(97);
fos.write('b');
fos.write('c');
//文件中:abc
//要写入字符串:转成byte数组
String s = "helloworld";
fos.write(s.getBytes());
//文件中:abchelloworld
复制代码
实现复制操作:
FileInputStream fis = new FileInputStream("d:\\aa.txt");
FileOutputStream fos = new FileOutputStream("d:\\bb.txt");
byte[] buf = new byte[1024];
int count=0;
while ((count=fis.read(buf))!=-1){
fos.write(buf,0,count);
}
fis.close();
fos.close();
复制代码
字节缓冲流
缓冲流:BufferedInputStream/BufferedOutputStream
继承了过滤流(FilterOutputStream/FilterInputStream),而过滤流又继承了文件字节流
内置了缓存区(8k)
作用:提高IO效率,减少访问磁盘次数;数据存储在缓存区,flush是将缓冲区的内容写入文件中,也可以直接close,后面实现缓冲区的类类似
字节输入缓冲流:
// 创建BufferedInputStream,需要传入一个底层流
FileInputStream fis = new FileInputStream("d:\\aa.txt");
BufferedInputStream bis = new BufferedInputStream(fis);
// 读取:类似,但是这个有缓冲区效率更高
int data = 0;
while ((data=bis.read())!=-1){
System.out.print((char)data);
}
//当然也可以和文件字节流一样,直接再创建一个缓冲区实现,一模一样但是因为内部缓冲区的存在效率更高
// 关闭这一个就行
bis.close();
//out:abchelloworld
复制代码
字节输出缓冲流:
FileOutputStream fos = new FileOutputStream("d://bb.txt");
BufferedOutputStream bos = new BufferedOutputStream(fos);
//转义符如换行等也可以写入也可以
String s = "hello\r\n";
bos.write(s.getBytes());//这是写入了缓冲区,没有写入文件
bos.flush();//需要刷新才能从缓冲区真正写入文件
bos.close();//当然close时自动flush
复制代码
对象流
对象流:ObjectOutputStream/ObjectInputStream
可以实现写入或者读取对象
增强了缓冲区功能
增强了读写8种基本数据类型和字符串的功能,不只是能读写int和byte
readObject()从流中读取一个对象;writeObject(Object obj)向流中写入一个对象
使用流传输对象的过程被称为序列化(写入)和反序列化(读取)
继承文件字节流,需要基于其才能创建
序列化和反序列化的对象类都必须实现Serializable接口(该接口本身没有任何方法仅仅标志该类可以序列化),而且其中关联的其他类也要实现该接口
可以在类中自定义生成常量private static final long serialVersionUID ,该常量为序列化版本号用来标识该类,即使是同一个类如果UID不同依旧被认为不是同一个类无法反序列化不同UID的类
使用transient修饰实现,表示该属性不需要序列化(例如:private transient int age )
静态属性不能序列化
序列化:
public class Student implements Serializable {}
//main:
FileOutputStream fos = new FileOutputStream("d://aa.txt");
ObjectOutputStream oos = new ObjectOutputStream(fos);
// 序列化
Student student = new Student();
oos.writeObject(student);
// oos.writeBoolean();写入boolean
// oos.writeBytes("hellow");//写入字符串
// 其他基本类型也有相应的办法写入
oos.close();
复制代码
反序列化:
FileInputStream fis = new FileInputStream("d://aa.txt");
ObjectInputStream ois = new ObjectInputStream(fis);
//读取存入的第一个对象
Student s = (Student)ois.readObject();
ois.close();
复制代码
序列化、反序列化多个对象时可以使用集合:
//序列化
FileOutputStream fos = new FileOutputStream("d://aa.txt");
ObjectOutputStream oos = new ObjectOutputStream(fos);
// 序列化
Student student = new Student();
Student student1 = new Student();
ArrayList<Student> arrayList = new ArrayList<>();
arrayList.add(student);
arrayList.add(student1);
oos.writeObject(arrayList);
oos.close();
//反序列化
FileInputStream fis = new FileInputStream("d://aa.txt");
ObjectInputStream ois = new ObjectInputStream(fis);
ArrayList<Student> arrayList = (ArrayList<Student>) ois.readObject();
ois.close();
复制代码
常见字符编码
字符流
问题的引出:
FileInputStream fis = new FileInputStream("d://aa.txt");
int data = 0;
while ((data=fis.read())!=-1){
System.out.println((char)data);
}
fis.close();
//上述代码当我们读取的文件中为中文时得到乱码。
//原因:我们是运用字节流一个一个字节读取文件,而本处的文件编码方式是utf-8(中文常见的编码方式),每个中文字占3个字节。当一个一个字节读取时出现乱码
复制代码
这种情况下我们需要字符流
字符流的父类(抽象类):
Reader(字符输入流)和Writer(字符输出流)
只能用于文本文件
文件字符流
FileReader和FileWriter
父类是InputStreamReader和OutputStreamWriter,而这两者的父类又是Reader和Writer
采用默认字符编码
FileReader:
FileReader fr = new FileReader("d://aa.txt");
//单个字符的读取
// int data = 0;
// while ((data=fr.read())!=-1){//读取一个字符
// System.out.println((char)data);
// }
//创建缓冲区读取
//缓冲区为2字符
char[] buf = new char[2];
int count = 0;
while ((count = fr.read(buf))!=-1){
System.out.println(new String(buf,0,count));
}
fr.close();
/*out:
你好
呀
复制代码
FileWriter:
FileWriter fw = new FileWriter("d://aa.txt");
fw.write("你好吗");
fw.flush();
fw.close();
复制代码
字符缓冲流
缓冲流:BufferedReader/BufferedWriter
高效读写
支持输入换行符
可以一次写一行、读一行
可以指定缓冲区大小,也可以默认(8K)
新建需要基础字符流(文件字符流)
继承Reader和Writer
读取:
FileReader fr = new FileReader("d://aa.txt");
BufferedReader br = new BufferedReader(fr);
// 和上面其他类类似的方法不再重写
// 特有的:读一行,末尾返回null
String s = null;
//一行一行的读取
while ((s=br.readLine())!=null){
System.out.println(s);
}
br.close();
//out:你好呀
复制代码
写入:
FileWriter fw = new FileWriter("d:\\aa.txt");
BufferedWriter bw = new BufferedWriter(fw);
// 写入
bw.write("好好学习");
// 写入一个换行windows:\r\n linux:\n
bw.newLine();
bw.write(",天天向上");
bw.close();
/*文件:
好好学习
,天天向上
复制代码
打印流
PrintWriter:
支持print()/println(),支持写入后换行
支持数据原样打印
继承Writer类
类似PrintStream,但是PrintWriter只能打印字符流,而PrintStream可以打印字节流
PrintWriter pw = new PrintWriter("d://aa.txt");
// 打印并且换行
pw.println(97);
pw.println(true);
//打印
pw.print("a");
pw.print("b");
pw.close();
/*文件中:
97
true
ab
复制代码
转换流
也叫桥转换流:InputStreamReader/OutputStreamWriter
可以将字节流转换为字符流
可设置字符的编码方式
继承字符流Reader/Writer
创建时需要基本的文件字符流
InputStreamReader:
FileInputStream fis = new FileInputStream("d://aa.txt");
InputStreamReader isr = new InputStreamReader(fis,"utf-8");
int data=0;
//一个一个字符读取
while ((data=isr.read())!=-1){
System.out.print((char)data);
}
isr.close();
复制代码
OutputStreamWriter:
FileOutputStream fos = new FileOutputStream("d://aa.txt");
OutputStreamWriter osw = new OutputStreamWriter(fos,"gbk");
for(int i=0;i<10;i++){
osw.write("我爱中国\r\n");
osw.flush();
}
osw.close();
复制代码
Prorerties
属性集合
特点:
存储属性名和属性值
属性名和属性值都是字符串类型
没有泛型
继承Hashtable集合,具体看集合那一章笔记
和流有关
// 分隔符
System.out.println("路径分割符"+File.pathSeparator);
System.out.println("名字分割符"+File.separator);
File file = new File("d://cc.txt");
// 判断文件是否存在
if(!file.exists()){
// 创建文件
boolean b = file.createNewFile();
System.out.println("创建结果:"+b);
}
if(file.exists()) {
// 直接删除
System.out.println("删除结果" + file.delete());
//使用jvm退出时删除
file.deleteOnExit();
}
// 获取文件信息
System.out.println("绝对路径"+file.getAbsolutePath());
//看你创建时传入的路径
System.out.println("获取路径"+file.getPath());
// 获取名称
System.out.println("获取名称"+file.getName());
// 获取父目录
System.out.println("获取父目录"+file.getParent());
// 获取文件长度
System.out.println("获取文件长度"+file.length());
// 文件创建时间
System.out.println("文件创建时间"+new Date(file.lastModified()).toString());
// 判断文件是否可写
System.out.println("是否可写"+file.canWrite());
// 是否是文件
System.out.println("是否是文件"+file.isFile());
// 是否隐藏
System.out.println("是否隐藏"+file.isHidden());
/*
路径分割符;
名字分割符\
创建结果:true
删除结果true
绝对路径d:\cc.txt
获取路径d:\cc.txt
获取名称cc.txt
获取父目录d:\
获取文件长度0
文件创建时间Thu Jan 01 08:00:00 CST 1970
是否可写false
是否是文件false
是否隐藏false
复制代码
File dir = new File("d://cc/aa");
// 判断文件是否存在
if(!dir.exists()){
// 创建文件夹:只能创建单集目录
// boolean b = dir.mkdir();
// 创建多级目录
boolean b = dir.mkdirs();
System.out.println("创建结果:"+b);
}
if(dir.exists()) {
// 直接删除:最下层且必须为空目录
System.out.println("删除结果" + dir.delete());
//使用jvm退出时删除,休眠结束后删除
dir.deleteOnExit();
Thread.sleep(5000);
}
// 获取文件信息
System.out.println("绝对路径"+dir.getAbsolutePath());
//看你创建时传入的路径
System.out.println("获取路径"+dir.getPath());
// 获取名称
System.out.println("获取名称"+dir.getName());
// 获取父目录
System.out.println("获取父目录"+dir.getParent());
// 文件创建时间
System.out.println("文件创建时间"+new Date(dir.lastModified()).toString());
// 判断是否是文件夹
System.out.println("是否可写"+dir.isDirectory());
// 是否隐藏
System.out.println("是否隐藏"+dir.isHidden());
// 遍历文件夹
String[] file = dir.list();
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
欢迎光临 IT评测·应用市场-qidao123.com (https://dis.qidao123.com/)
Powered by Discuz! X3.4