我们一直都是在操作文件或者文件夹,并没有给文件中写任何数据。现在我们就要开始给文件中写数据,或者读取文件中的数据。

字节输出流写入数据到文件

写入数据的原理(内存-->硬盘)

    java程序-->JVM(java虚拟机)-->OS(操作系统)-->OS调用写数据的方法-->把数据写入到文件中

字节输出流的使用步骤(重点):

    1.创建一个FileOutputStream对象,构造方法中传递写入数据的目的地

    2.调用FileOutputStream对象中的方法write,把数据写入到文件中

    3.释放资源(流使用会占用一定的内存,使用完毕要把内存清空,提供程序的效率)

文件存储的原理和记事本打开文件的原理 

字节输出流OutputStream

OutputStream此抽象类,是表示输出字节流的所有类的超类。操作的数据都是字节,定义了输出字节流的基本共性功能方法。

java.io.OutputStream:字节输出流

    此抽象类是表示输出字节流的所有类的超类。 

定义了一些子类共性的成员方法:

- public void close() :关闭此输出流并释放与此流相关联的任何系统资源。
    - public void flush() :刷新此输出流并强制任何缓冲的输出字节被写出。
    - public void write(byte[] b):将 b.length字节从指定的字节数组写入此输出流。
    - public void write(byte[] b, int off, int len) :从指定的字节数组写入 len字节,从偏移量 off开始输出到此输出流。
    - public abstract void write(int b) :将指定的字节输出流。
 
java.io.FileOutputStream extends OutputStream
FileOutputStream:文件字节输出流

作用:把内存中的数据写入到硬盘的文件中 

构造方法:

FileOutputStream(String name)创建一个向具有指定名称的文件中写入数据的输出文件流。
    FileOutputStream(File file) 创建一个向指定 File 对象表示的文件中写入数据的文件输出流。

    参数:写入数据的目的

        String name:目的地是一个文件的路径

        File file:目的地是一个文件

    构造方法的作用:

        1.创建一个FileOutputStream对象

        2.会根据构造方法中传递的文件/文件路径,创建一个空的文件

        3.会把FileOutputStream对象指向创建好的文件

 输出流中定义都是写write方法,如下图:

Android 字节流文件 字节流输出文件_输入流

虽然参数为int类型四个字节,但是只会保留一个字节的信息写出

当完成流的操作时,必须调用close()方法,释放系统资源。

FileOutputStream类

OutputStream有很多子类,其中子类FileOutputStream可用来写入数据到文件。

FileOutputStream类,即文件输出流,是用于将数据写入 File的输出流。

FileOutputStream类写入数据到文件中

将数据写到文件中,代码演示:

public class FileOutputStreamDemo {
	public static void main(String[] args) throws IOException {
		//需求:将数据写入到文件中。
		//创建存储数据的文件。
		File file = new File("c:\\file.txt");
		//创建一个用于操作文件的字节输出流对象。一创建就必须明确数据存储目的地。
		//输出流目的是文件,会自动创建。如果文件存在,则覆盖。
		FileOutputStream fos = new FileOutputStream(file);
		//调用父类中的write方法。
		byte[] data = "abcde".getBytes();
		fos.write(data);
		//关闭流资源。
		fos.close();
	}
}

给文件中续写和换行

我们直接new FileOutputStream(file)这样创建对象,写入数据,会覆盖原有的文件,那么我们想在原有的文件中续写内容怎么办呢?

继续查阅FileOutputStream的API。发现在FileOutputStream的构造函数中,可以接受一个boolean类型的值,如果值true,就会在文件末位继续添加。

回车符\r和换行符\n :

回车符:回到一行的开头(return)

换行符:下一行(newline)

 系统中的换行:

Windows系统里,每行结尾是 回车+换行 ,即\r\n;

Unix系统里,每行结尾只有换行,即\n;

Mac系统里,每行结尾是回车,即\r。从 Mac OS X开始与Linux统一

 给文件中续写数据和换行,代码演示:

public static void main(String[] args) throws Exception {
		File file = new File("c:\\file.txt");
		FileOutputStream fos = new FileOutputStream(file, true);
		String str = "\r\n"+"itcast";
		fos.write(str.getBytes());
		fos.close();
	}

字节输入流InputStream

通过前面的学习,我们可以把内存中的数据写出到文件中,那如何想把内存中的数据读到内存中,我们通过InputStream可以实现。InputStream此抽象类,是表示字节输入流的所有类的超类。,定义了字节输入流的基本共性功能方法。

java.io.InputStream:字节输入流

此抽象类是表示字节输入流的所有类的超类。

 定义了所有子类共性的方法:

int read()从输入流中读取数据的下一个字节。
     int read(byte[] b) 从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b 中。
     void close() 关闭此输入流并释放与该流关联的所有系统资源。

 java.io.FileInputStream extends InputStream

FileInputStream:文件字节输入流

作用:把硬盘文件中的数据,读取到内存中使用

 构造方法:

FileInputStream(String name)
    FileInputStream(File file)

    参数:读取文件的数据源

        String name:文件的路径

        File file:文件

    构造方法的作用:

        1.会创建一个FileInputStream对象

        2.会把FileInputStream对象指定构造方法中要读取的文件

int read():读取一个字节并返回,没有字节返回-1.
int read(byte[]): 读取一定量的字节数,并存储到字节数组中,返回读取到的字节数。

字节输入流一次读取多个字节

字节输入流一次读取多个字节的方法:

    int read(byte[] b) 从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b 中。

明确两件事情:

    1.方法的参数byte[]的作用?

        起到缓冲作用,存储每次读取到的多个字节

        数组的长度一把定义为1024(1kb)或者1024的整数倍

    2.方法的返回值int是什么?

        每次读取的有效字节个数 

String类的构造方法

String(byte[] bytes) :把字节数组转换为字符串
    String(byte[] bytes, int offset, int length) 把字节数组的一部分转换为字符串 offset:数组的开始索引 length:转换的字节个数

FileInputStream类

InputStream有很多子类,其中子类FileInputStream可用来读取文件内容。
FileInputStream 从文件系统中的某个文件中获得输入字节。

FileInputStream类读取数据read方法

在读取文件中的数据时,调用read方法,实现从文件中读取数据

read方法,每次可以读取一个字节的数据,提升为int类型,读取到文件末尾,返回-1 

读取数据read(byte[])方法

在读取文件中的数据时,调用read方法,每次只能读取一个,太麻烦了,于是我们可以定义数组作为临时的存储容器,这时可以调用重载的read方法,一次可以读取多个字符。 

每次读取b的长度个字节到数组中,返回读取到的有效字节个数,读取到末尾时,返回-1,

public class FileInputStreamDemo2 {
	public static void main(String[] args) throws IOException {
		/*
		 * 演示第二个读取方法, read(byte[]);
		 */
		File file = new File("c:\\file.txt");
		// 创建一个字节输入流对象,必须明确数据源,其实就是创建字节读取流和数据源相关联。
		FileInputStream fis = new FileInputStream(file);		
		//创建一个字节数组。
		byte[] buf = new byte[1024];//长度可以定义成1024的整数倍。		
		int len = 0;
		while((len=fis.read(buf))!=-1){
			System.out.println(new String(buf,0,len));
		}
		fis.close();
	}
}

缓冲数组方式复制文件

既然会了文件的读和写操作了,那么我们就要在这个基础上进行更为复杂的操作。使用读写操作完成文件的复制。

复制文件

原理:读取一个已有的数据,并将这些读到的数据写入到另一个文件中。

文件复制的步骤:

    1.创建一个字节输入流对象,构造方法中绑定要读取的数据源

    2.创建一个字节输出流对象,构造方法中绑定要写入的目的地

    3.使用字节输入流对象中的方法read读取文件

    4.使用字节输出流中的方法write,把读取到的字节写入到目的地的文件中

    5.释放资源

public class CopyFileTest {
	public static void main(String[] args) throws IOException {
		//1,明确源和目的。
		File srcFile = new File("c:\\YesDir\test.JPG");
		File destFile = new File("copyTest.JPG");
		
		//2,明确字节流 输入流和源相关联,输出流和目的关联。
		FileInputStream fis = new FileInputStream(srcFile);
		FileOutputStream fos = new FileOutputStream(destFile);
		
		//3, 使用输入流的读取方法读取字节,并将字节写入到目的中。
		int ch = 0;
		while((ch=fis.read())!=-1){
			fos.write(ch);
		}
		//4,关闭资源。
		fos.close();
		fis.close();
	}
}

上述代码输入流和输出流之间是通过ch这个变量进行数据交换的。

上述复制文件有个问题,每次都从源文件读取一个,然后在写到指定文件,接着再读取一个字符,然后再写一个,一直这样下去。效率极低。

 上述代码复制文件效率太低了,并且频繁的从文件读数据,和写数据,能不能一次多把文件中多个数据都读进内容中,然后在一次写出去,这样的速度一定会比前面代码速度快。

public class CopyFileByBufferTest {
	public static void main(String[] args) throws IOException {
		File srcFile = new File("c:\\YesDir\test.JPG");
		File destFile = new File("copyTest.JPG");
		// 明确字节流 输入流和源相关联,输出流和目的关联。
		FileInputStream fis = new FileInputStream(srcFile);
		FileOutputStream fos = new FileOutputStream(destFile);
		//定义一个缓冲区。
		byte[] buf = new byte[1024];
		int len = 0;
		while ((len = fis.read(buf)) != -1) {
			fos.write(buf, 0, len);// 将数组中的指定长度的数据写入到输出流中。
		}
		// 关闭资源。
		fos.close();
		fis.close();
	}
}