一、字节数组流
之前使用输入输出流的操作的对象是文件,而这里字节数组流操作的对象是内存,内存可以看做是一个字节数组。
使用字节数组流读写就可以看做是从内存A到内存B的读写,对象时内存即字节数组。
1.1构造方法
ByteArrayOutputStream()//创建一个字符数组输出流
ByteArrayInputStream(byte[] buf)//创建一个字节数组输入流,这里面的buff可看做内存对象即字符数组
2.2主要方法
ByteArrayInputStream:
read(byte[] bu)//读取输入流中的数据,放入bu中。
ByteArrayOutputStream:
public void write(byte[] bu)//将bu(可看做内存对象数据),写入输出流。
byte[] toByteArray()//创建一个新的字节数组,并将输出流中数据放入其中。
2.3例子
import java.io.BufferedInputStream; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.InputStream; public class ByteArrayStream { public static void main(String[] args) { // TODO Auto-generated method stub read(write()); } public static void read(byte[] by){ int len = 0; //将内存对象B放入输入流 InputStream r = new BufferedInputStream(new ByteArrayInputStream(by));//把这里的by看做内存对象 byte[] flush = new byte[1024];//把flush看做另外一个内存对象 try { //将内存对象B与输入流关联,来后通过输入流读取对象B数据放入flush中 while(-1 != (len = r.read(flush))){//读取输入流中的数据放入flush,即读取内存对象B中的数据 System.out.println(new String(flush,0,len));//打印出读取的数据 } } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } public static byte[] write(){ byte[] by = null;//内存对象B byte []info = "字节数组流".getBytes();//内存对象A ByteArrayOutputStream bos = new ByteArrayOutputStream(); try { bos.write(info);//将内存A块内容写入输出流 by = bos.toByteArray();//内存对象B接收输出流中的数据 } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } return by;//返回内存对象B } }
运行结果:
字节数组流
wtire:首先将内存A的数据写入输出流,然后内存B获取输入流中的数据,最后将内存B作为返回值。
read:将内存B放入输入流,然后通过输入流读取内存B中的数据放入flush中,然后将flush转为字符打印出来。
这里在一个程序中运行看着有点奇怪而且不好理解,我们就把内存A内存B看着是两台电脑各自内存,这样便于理解一些。
二、数据流
采用数据流进行读写可以允许将原始java数据类型写入流中,即既保持了数据也保存了数据类型。
2.1构造方法
DataInputStream(InputStream in)
DataOutputStream(OutStream out)
2.2主要方法
void writeInt(int v)//写入int型数据
void writeLong(long v)//写入long型数据
void writeUTF(String str)//将字符串以UTF-8格式写入输出流
int readInt()//读取Int型数据并返回
long readLong()//读取long型数据并返回
String readUTF()//读取UTF-8转换的字符串
基本操作方法与文件读写无太大差别,只是加上了数据类型,这些方法时DataStream独有的方法,故调用它们要避免多态。
2.3例子
import java.io.BufferedInputStream; import java.io.BufferedOutputStream; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; public class DateStream { public static void main(String[] args) throws IOException { // TODO Auto-generated method stub write(); read(); } public static void read() throws IOException{ DataInputStream dis = new DataInputStream(//数据流 new BufferedInputStream(//缓冲流 new FileInputStream( new File("F:\依风\Desktop\data.txt") ) ) ); //注意:这里读取类型的顺序要和写入时的类型顺序对应,不然会出现错误。 long lo = dis.readLong();//将不同类型用对应方法读出。 int in = dis.readInt(); String str = dis.readUTF(); dis.close(); System.out.println(lo + "-->" + in +"-->" + str); } public static void write() throws IOException{ long lo = 100L; int in = 21; String str = "DataStream"; DataOutputStream dos = new DataOutputStream(//数据流 new BufferedOutputStream(//缓冲流 new FileOutputStream( new File("F:\依风\Desktop\data.txt") ) ) ); dos.writeLong(lo);//将不同类型用对应的方法写入 dos.writeInt(in); dos.writeUTF(str); dos.flush(); dos.close(); } }
运行结果:
100-->21-->DataStream
这里一开始往文件中写入的是long型,然后是int类型最后是String
那么读取时也应该先读long,然后int最后String。
如果写入和读取的顺序不一致可能会导致读取的数据有误,也可以出现EOFException。
这里写入的数据不是给人类看的,而是给机器识别的。我可以打开写入的文件会发现里面会有乱码。
三、对象流
使用对象流,可以写入、读出对象的信息以及属性属性。对象流和之前输入输出流功能类似,
不过是将之前的文件换成了对象,这里的对象时我们通过new创建的对象。
将对象写如文件实质是是将对象转换为字节序列,然后将字节序列写入文件,这个称为序列化。
读取字节序列然后转换为对象就称为反序列化。
对象被序列化必须实现java.io.Serializable接口,改接口表示可以序列化。对象的类没有实现这个接口会出现一次次。
对象中某些属性不想被序列化可以加transient修饰,transient修饰就代表这个属性不参加序列化。
3.1构造方法
Public ObjectOutputStream(OutputStream out)
Public ObjectInputStream(InputStream in)
3.2主要方法
Object readObject()//从输入流中读取对象
void writeObject(Object obj)//将obj写入输出流
3.3例子
import java.io.BufferedInputStream; import java.io.BufferedOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; public class ObjectStream { public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException{ read(write()); } //将对象写入指定文件,并返回指定文件路径信息 public static String write() throws FileNotFoundException, IOException{ String filePath = "F:\依风\Desktop\ObjectStream.txt"; ObjectOutputStream oos = new ObjectOutputStream(//使用独有方法,不发生多态 new BufferedOutputStream( new FileOutputStream( new File(filePath)))); oos.writeObject(new Employ("hcf",4000));//传入对象 oos.flush(); oos.close(); return filePath; } //接受指定文件路径信息,并读取文件中的对象 public static void read(String filePath) throws FileNotFoundException, IOException, ClassNotFoundException{ ObjectInputStream ois = new ObjectInputStream( new BufferedInputStream( new FileInputStream( new File(filePath)))); Employ em = (Employ)ois.readObject();//将读取的对象强制转型 System.out.println(em.getName() + ":" + em.getMoney());//输出对象信息 } } class Employ implements java.io.Serializable{//必须实现接口 private transient String name;//姓名没有被序列化 private int money; public Employ(){} public Employ(String name,int money){ setName(name); setMoney(money); } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getMoney() { return money; } public void setMoney(int money) { this.money = money; } }
运行结果: null:4000
我们可以看到没有被序列化的属性显示为null
比如一个矩形类中有长、宽属性,还有面积属性,
那么序列化时可以不序列化面积属性,因为面积属性可以通过长宽计算出来。
四、打印流
打印流(PraintStream)类似输出流,但打印流使用方便而且可以打印出任何类型的数据。
想我们平常使用很多的System.out.println()就是调用打印流中的println()方法。
4.1构造函数
PrintStream(OutputStream out)//创建打印流
PrintStream(OutputStream out, boolean autoFlush)//创建打印流,autoFlush为true代表自动刷新缓冲区
PrintStream(File fileName, String csn)//创建指定文件的打印流,csn为编码方式
4.3主要方法
print(......);//打印参数内容,几乎可为任何形式参数,末尾不加回车。
println(......)//打印参数内容,末尾加回车。
4.4例子
import java.io.BufferedOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.PrintStream; import java.io.UnsupportedEncodingException; import java.util.Scanner; public class Print { public static void main(String[] args) throws FileNotFoundException, UnsupportedEncodingException{ PrintStream ps = new PrintStream(//常规初始化 new BufferedOutputStream( new FileOutputStream( new File("F:\依风\Desktop\print.txt")))); PrintStream pc = new PrintStream("F:\依风\Desktop\printCsn.txt","GBK");//指定编码 PrintStream pf = new PrintStream(//自动刷新 new BufferedOutputStream( new FileOutputStream( new File("F:\依风\Desktop\printFlush.txt"))),true); pf.println("autoFlush"); pf.close(); pc.print("GBK"); pc.flush(); pc.close(); ps.print("print "+1.3); ps.flush(); ps.close(); } }
运行结果:
4.5 Sytem.out和 System.in
我们平常调用System.out.println()函数打印,打印的内容是显示在控制台上的,System.out也是一个打印流。
而这个打印流关联的就是控制台,可以把控制台也看做一个文件,平常打印出来的数据就是往控制台这个文件打印。
那么我们能否将这个文件换成别的文件呢,即调用System.out.println()不是向控制台输出信息,而是向我们指定的文件输入信息。
答案是可以的,System里面提供了一个方法System.SetOut(PrintStream out)方法,可以设置System.out。
可以看到参数是一个打印流。
设置指定的参数后,Sytem.out.println()就会将内容打印到指定的打印流中。
public class Print { public static void main(String[] args) throws FileNotFoundException, UnsupportedEncodingException{ System.setOut(new PrintStream(new FileOutputStream(new File("F:\依风\Desktop\print.txt")))); System.out.println("setOut"); } }
还有System.in,这个比较常用的用法就是作为Scanner的参数,表示从键盘输入。
System.in其实就是一个输入流(InputStream),而System.in默认是键盘,这里可以把键盘也看做一个文件,就是从键盘这个文件读取内容。
那么同样的,我们也可以将System.in修改为其他文件。
就需要用到System.setIn(InputStream in).这时就是读取设置之后的输入流
import java.io.BufferedInputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.PrintStream; import java.io.UnsupportedEncodingException; import java.util.Scanner; public class Print { public static void main(String[] args) throws FileNotFoundException, UnsupportedEncodingException{ //重定向了System.in将原来的将键盘输入更改为其他输入 System.setIn(new BufferedInputStream(new FileInputStream(new File("F:\依风\Desktop\print.txt")))); Scanner si = new Scanner(System.in); System.out.println(si.nextLine());//读取一行 } }
运行结果:
setOut//根据文件内容决定
那么我么修改了Sytem.in(键盘输入)和System.out(控制台输出)为其他的文件后是否可以还原它们,当然是可以的,之前说了键盘、控制台都是文件
(在windows中一切都可以看做是文件),现在的关键就是要找到代表控制台可键盘的文件,然后再次Set就可以了。
我们来看下源码:
其中FileDescriptor.in就代表键盘,....out就代表控制台,
public class Print { public static void main(String[] args) throws FileNotFoundException, UnsupportedEncodingException{ //输出重定向为指定文件 System.setOut(new PrintStream(new FileOutputStream(new File("F:\依风\Desktop\print.txt")))); System.out.println("setSystem.outAndSystem.in"); System.out.close(); //输出重定向为控制台 System.setOut(new PrintStream(new FileOutputStream(FileDescriptor.out))); //输入重定向为指定文件 System.setIn(new BufferedInputStream(new FileInputStream(new File("F:\依风\Desktop\print.txt")))); Scanner si = new Scanner(System.in); System.out.println(si.nextLine());//内容输出控制台 } }
运行结果:
setSystem.outAndSystem.in
还有一个System.err也是一个打印流,这个主要用于打印出错误信息。比如:
这些红色的错误信息就是err打印出来的。
import java.io.BufferedInputStream; import java.io.File; import java.io.FileDescriptor; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.PrintStream; import java.io.UnsupportedEncodingException; import java.util.Scanner; public class Print { public static void main(String[] args) throws FileNotFoundException, UnsupportedEncodingException{ System.err.println("错误信息"); } }
运行结果: