一、字节数组流

  之前使用输入输出流的操作的对象是文件,而这里字节数组流操作的对象是内存,内存可以看做是一个字节数组。

  使用字节数组流读写就可以看做是从内存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\n"+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("错误信息");
}
}

运行结果:

最新文章

  1. IOS 杂笔-12(类别de巧用 有便于Frame的操作)
  2. 框架集(Framesets)
  3. JQuery 实现锚点链接之间的平滑滚动
  4. 如何成为python高手
  5. 微信小程序 开发 微信开发者工具 快捷键
  6. iOS10 UI教程视图和子视图的可见性
  7. Adapter适配器
  8. shell脚本中的[]/[[]]区别
  9. LNMP下wordpress无法切换主题,只显示当前主题解决方法
  10. *MySQL卸载之后无法重装,卡在Apply security settings:Error Nr.1045
  11. <一> MVC - HtmlHelper
  12. C语言学习笔记——堆和栈——未整理
  13. ARM编译器4字节对齐
  14. IFrame父页面和子页面的交互
  15. Android开发之漫漫长途 ⅥI——Android消息机制(Looper Handler MessageQueue Message)
  16. 关于sqlmap使用手册
  17. 步步深入:MySQL架构总览->查询执行流程->SQL解析顺序(转)
  18. 使用jsdelivr访问github资源
  19. SQL Server 锁机制
  20. 模拟页面获取的php数据(三)

热门文章

  1. CSS3学习笔记之loading动画
  2. C# 序列化原因 (转)
  3. 51nod 1040 最大公约数之和
  4. 用 C# 代码如何实现让你的电脑关机,重启,注销,锁定,休眠,睡眠
  5. eclipse导入java和android sdk源码,帮助文档
  6. 转载:超级强大的vim配置(vimplus)--续集
  7. JAVA -- JDK JRE JAR
  8. HTML5-坦克大战一完成坦克上下左右移动的功能(一)
  9. Solidity 文档--第一章:智能合约入门
  10. Combo Box的简单使用(Win32)