java输入流和输出流文件应该建在什么位置

Python016

java输入流和输出流文件应该建在什么位置,第1张

在Java程序中,要想从文件中读取数据,需要在程序和文件之间建立一条数据输入的通道,这样程序就可以从文件中读取数据了;反之,如果要在Java程序中把数据写入到文件中,也需要在程序和文件之间建立一条数据输出的通道。当程序创建输入流对象时,Java会自动建立这个数据输入通道,而创建输出流时,Java也会自动建立这个数据输出通道。如下图所示:

Java程序读取与写入数据

输入流是从文件读取数据,是一个拉取数据的过程;输出流是将数据写入到文件,是一个推送数据的过程。

为了便于理解输入输出流,前面都是以文件为数据源来讨论的。其实,Java的输入与输出流支持任何数据源的读取与写入,包括键盘、文件、网络、数据库等数据源。

输入流和输出流按读取和写入的数据单位可分为字节流和字符流,字节流是以字节为单位传输数据的流,字符流是以字符为单位传输数据的流。

Java所提供的输入流和输出流类封装在Java.io包中

Java 操作文件包括了两个层面:

在文件系统的层面上来操作文件,包括创建文件、删除文件、创建目录、拷贝文件等等。

操作文件里保存的数据,包括读文件、写文件。

1. 在文件系统层面操作文件

1.1 File 类介绍

Java 中通过 java.io.File 类来对一个文件(包括目录)进行抽象的描述。不过有 File 对象,并不代表对应的文件真实存在。

1.2 File 常见属性

属性 修饰符及类型 说明

pathSeparator static String 依赖于系统的路径分隔符,String 类型的表示

pathSeparator static char 依赖于系统的路径分隔符,char 类型的表示

1.3 File 构造方法

构造方法 说明

File(File parent, String child) 根据父目录+孩子文件路径,创建一个新的 File 实例

File(String pathname) 根据文件路径创建一个新的 File 实例,路径可以是绝对路径或者相对路径

File(String parent, String child) 根据父目录路径+孩子文件路径,创建一个新的 File 实例

1.4 File 常用方法

方法 返回值类型 说明 注意

getParent() String 返回 File 对象的父目录文件路径

getName() String 返回 File 对象的纯文件名称

getPath() String 返回 File 对象的文件路径

getAbsolutePath() String 返回 File 对象的绝对路径

getCanonicalPath() String 返回 File 对象的修饰过的绝对路径 如果绝对路径的值为:d:/././test.txt,那么修饰过的绝对路径就为:d:/test.txt。

exists() boolean 判断 File 对象描述的文件是否真实存在

isDirectory() boolean 判断 File 对象代表的文件是否是一个目录

isFile() boolean 判断 File 对象代表的文件是否是一个普通文件

createNewFile() boolean 根据 File 对象,自动创建一个空文件。成功创建后返回 true 如果文件路径不正确或者没有权限,则可能创建失败。

delete() boolean 根据 File 对象,删除该文件。成功删除后返回 true 如果文件路径不正确或者没有权限,则可能删除失败。

deleteOnExit() void 根据 File 对象,标注文件将被删除,删除动作会到 JVM 运行结束时才会进行

list() String[] 返回 File 对象代表的目录下的所有文件名

listFiles() File[] 返回 File 对象代表的目录下的所有文件,以 File 对象表示

mkdir() boolean 创建 File 对象代表的目录

mkdirs() boolean 创建 File 对象代表的目录,如果必要,会创建中间目录

renameTo(File dext) boolean 重命名文件或移动文件,当目录为空时也可以重命名目录

canRead() boolean 判断用户是否对文件有可读权限

canWrite() boolean 判断用户是否对文件有可写权限

2. 操作文件里的数据

文件内容的操作包括读文件和写文件,而对于这两种操作 Java 又提供了两种不同的类:

读类 写类 说明

InputStream OutputStream 字节流,针对二进制文件进行读写,操作基本单位是字节

Reader Writer 字符流,针对文本文件进行读写,操作基本单位是字符

字节流: 它处理单元为1个字节,操作字节和字节数组,存储的是二进制文件,如果是音频文件、图片、歌曲,就用字节流。

字符流: 它处理的单元为2个字节的 Unicode 字符,分别操作字符、字符数组或字符串,字符流是由 Java 虚拟机将字节转化为2个字节的 Unicode 字符为单位的字符而成的,如果是关系到中文(文本)的,用字符流更好。

所有文件都是以字节的形式储存,在磁盘上保留的并不是文件的字符而是先把字符编码成字节,再储存这些字节到磁盘。在读取文件(特别是文本文件)时,也是一个字节一个字节地读取以形成字节序列。

字节流可用于任何类型的对象,包括二进制对象,而字符流只能处理字符或者字符串。

字节流提供了处理任何类型的 IO 操作的功能,但它不能直接处理 Unicode 字符,而字符流就可以。

字节流在操作的时候本身不会用到缓冲区的,是与文件本身直接操作的;而字符流在操作的时候会使用到缓冲区的。

其中 InputStream/OutputStream 和 Reader/Writer 都实现了 Closeable 接口,所以可以不主动通过 close 方法去关闭该资源,而是通过代码可读性更高的 try-with-resource 操作在 try 中的内容结束后,无论是否有异常都关闭该资源。

使用 OutputStream 或 writer 打开文件默认会清空之前的文件内容

2.1 操作字节流数据

2.1.1 InputStream

InputStream 是一个抽象类,使用时需要使用具体的实现类。InputStream 的实现类很多,针对不同的输入设备都可以对应一个实现类,其中针对文件读取的实现类是 FileInputStream。

FileInputStream 构造方法:

构造方法 说明

FileInputStream(File file) 利用 File 构造对应文件输入流

FileInputStream(String name) 利用文件路径构造对应文件输入流

InputStream 核心方法:

方法 返回值类型 说明

read() int 一次读取一个字节的数据,如果返回 -1 表示文件读完了

read(byte[] b) int 一次最多读取 b.length 字节的数据到 b 中,返回值为实际读到的字节数量,如果返回 -1 表示文件读完了

read(byte[] b, int off, int len) int 一次最多读取 len 个字节到 b 中,并从 off 的位置开始存放,返回值为实际读到的字节数量,如果返回 -1 表示文件读完了

close() void 关闭字节流

为什么 read 方法的返回值是 Int 类型?

对于 read 方法的返回值,当读取的是数字和英文时,返回值是对应的 ascii 码;当读取的是汉字时,返回的是汉字的机内码,比如使用的是 GBK 编码方式,返回的就是 GBK 的内部编码。read 方法每次从 inputstream 中读取一个字节,而一个字节是8位,但当最高位是1的时候,二进制对应的 ascii 码值就是复数,而 ascii 是没有负数的,所以用 byte 类型表示不了。并且 11111111 对应的十进制值是 -1,而返回 -1 表示文件读取完毕了,则后面的数据就不会读了,因此 byte 类型不适合,int 类型则可以解决这个问题。

示例代码: 使用 read() 方法读取文件内容

public class Demo1 {

public static void main(String[] args) throws IOException {

File file = new File("./test.txt")

StringBuilder sb = new StringBuilder()

try (InputStream inputStream = new FileInputStream(file)){

while(true) {

int b = inputStream.read()

if(b == -1){

break

}

sb.append((char) b)

}

} catch (IOException e) {

e.printStackTrace()

}

System.out.println(sb.toString())

}

}

登录后复制

示例代码: 使用 read(byte[] b) 方法读取文件内容

public class Demo1 {

public static void main(String[] args) throws IOException {

File file = new File("./test.txt")

byte[] buffer = new byte[1024]

try (InputStream inputStream = new FileInputStream(file)){

while(true) {

int len = inputStream.read(buffer)

if(len == -1){

break

}

String s = new String(buffer, 0, len, "utf-8")

System.out.println(s)

}

} catch (IOException e) {

e.printStackTrace()

}

}

}

登录后复制

2.1.2 OutputStream

OutputStream 是一个抽象类,使用时需要使用具体的实现类。OutputStream 的实现类很多,针对不同的输出设备都可以对应一个实现类,其中针对文件读取的实现类是 FileOutputStream。

FileOutputStream 构造方法:

构造方法 说明

FileOutputStream(File file) 利用 File 构造对应文件输出流

FileOutputStream(String name) 利用文件路径构造对应文件输出流

FileOutputStream(File file, boolean append) append 表示是否追加数据到文件的末尾,为 true 表示追加,为 false 表示情空之前的内容重新加入

FileOutputStream(String name, boolean append) append 表示是否追加数据到文件的末尾,为 true 表示追加,为 false 表示情空之前的内容重新加入

OutputStream 核心方法:

方法 返回值类型 说明

write() int 一次写入一个字节的数据

write(byte[] b) int 一次最多写入 b.length 字节的数据到 b 中,返回值为实际写入的字节数量

write(byte[] b, int off, int len) int 一次最多写入 len 个字节到 b 中,并从 off 的位置开始存放,返回值为实际写入到的字节数量

close() void 关闭字节流

示例代码:

public class Demo1{

public static void main(String[] args) throws IOException {

File in = new File("./in.txt")

File out = new File("./out.txt")

byte[] buffer = new byte[1024]

try (InputStream inputStream = new FileInputStream(in)

OutputStream outputStream = new FileOutputStream(out)) {

while (true) {

int len = inputStream.read(buffer)

if (len == -1) {

break

}

outputStream.write(buffer, 0, len)

}

} catch (IOException e) {

e.printStackTrace()

}

}

}

登录后复制

2.1.3 BufferedInputStream

BufferedInputStream 也是 InputStream 的具体实现类,相比于 FileInputStream 优化了系统调用,即通俗解释为:将数据保存了缓存区,但是减少磁盘 IO 并不是 BufferedInputStream 来做的,而是 OS 来做的。OS 根据局部性原理,会预读部分的数据到内存缓存区,这样下次 IO 如果读取数据在缓存命中了,就不需要等待磁盘的寻址,而是直接返回数据,效率就会快很多。

2.1.4 BufferedOutputStream

BufferedOutPutStream 也是 OutputStream 的具体实现类,相比于 FileOutputStream 优化了系统调优,即每次写数据的时候,都会将数据放入缓存区中,等缓存区满了之后,才会一次性写入到磁盘中,大大提高了效率。

2.2 操作字符流数据

2.2.1 Reader

Reader 是实现字符输入流的一种类型,其本身属于一个抽象类,使用时需要使用具体的实现类。Reader 的实现类很多,针对不同的输入设备都可以对应一个实现类,其中针对文件读取的实现类是 FileReader。Reader 的使用方式和 InputStream 相仿。

示例代码:

public class Demo2 {

public static void main(String[] args) throws FileNotFoundException {

File file = new File("./test.txt")

StringBuilder sb = new StringBuilder()

try (Reader reader = new FileReader(file)) {

while (true) {

int read = reader.read()

if (read == -1) {

break

}

sb.append((char) read)

}

} catch (IOException e) {

e.printStackTrace()

}

System.out.println(sb.toString())

}

}

登录后复制

2.2.2 Writer

Writer 是实现字符输出流的一种类型,其本身属于一个抽象类,使用时需要使用具体的实现类。Writer 的实现类很多,针对不同的输出设备都可以对应一个实现类,其中针对文件读取的实现类是 FileWriter。Writer 的使用方式和 OutputStream 相仿。

FileWriter 构造方法:

构造方法 说明

FileWriter(File file) 利用 File 构造对应文件输出流

FileWriter(String name) 利用文件路径构造对应文件输出流

FileWriter(File file, boolean append) append 表示是否追加数据到文件的末尾,为 true 表示追加,为 false 表示情空之前的内容重新加入

FileWriter(String name, boolean append) append 表示是否追加数据到文件的末尾,为 true 表示追加,为 false 表示情空之前的内容重新加入

示例代码:

public class Demo2 {

public static void main(String[] args) throws FileNotFoundException {

File in = new File("./in.txt")

File out = new File("./out.txt")

try (Reader reader = new FileReader(in)

Writer writer = new FileWriter(out)) {

while (true) {

int read = reader.read()

if (read == -1) {

break

}

writer.write(read)

}

} catch (IOException e) {

e.printStackTrace()

}

}

}

登录后复制

2.2.3 BufferedReader

BufferedReader 也是 Reader 的具体实现类,相比于 Reader 优化了系统调用,即通俗解释为:将数据保存了缓存区,但是减少磁盘 IO 并不是 BufferedReader 来做的,而是 OS 来做的。OS 根据局部性原理,会预读部分的数据到内存缓存区,这样下次 IO 如果读取数据在缓存命中了,就不需要等待磁盘的寻址,而是直接返回数据,效率就会快很多。

2.2.4 BufferedWriter

BufferedWriter 也是 Writer 的具体实现类,相比于 Writer 优化了系统调优,即每次写数据的时候,都会将数据放入缓存区中,等缓存区满了之后,才会一次性写入到磁盘中,大大提高了效率。

2.3 Scanner 和 PrintWriter

Java 标准库中提供的 Scanner 和 PrintWriter 类能够像文件中读取文本数据和写入文本数据。

2.3.1 Scanner

java.util.Scanner 类常用来从控制台中读取字符串和基本类型的数值。Scanner 可以将输入的值以空白字符分割标记。

Scanner 的构造方法:

构造方法 说明

Scanner(Readable source) 创建一个 Scanner,从指定的可读资源中扫描标记,包括文件、文件路径、字符串等。

Scanner(InputStream source) 创建一个 Scanner,从指定的字节输入流中扫把标记。

Scanner 常用方法:

方法 返回值类型 说明

hasNext() boolean 如果 Scanner 还要更多的数据可读取,则返回 true

next() String 从 Scanner 中读取下一个标记作为字符串返回

nextLine() String 从 Scanner 中读取一行,以换行符结束

close() void 关闭 Scanner

System.in 为系统自带的标准输入流,该流是一直打开并准备提供输入数据。通常,这个流对应于键盘输入或由主机环境或用户指定的另一个输入源。

使用 Scanner 没有必要关闭输入文件,但是关闭能够释放文件占有的资源。

示例代码:

public class Demo3 {

public static void main(String[] args) {

try (Scanner scanner = new Scanner(new File("./test.txt"))) {

while (scanner.hasNext()) {

String s = scanner.next()

System.out.println(s)

}

} catch (IOException e) {

e.printStackTrace()

}

}

}

登录后复制

2.3.2 PrintWriter

java.io.PrintWriter 类可用来创建一个文件并向文本文件中写入数据。通常使用 System.out.println 向控制台输入文本。

PrintWriter 构造方法:

构造方法 说明

PrintWriter(Writer out) 为指定的字符输入流创建一个 PrintWriter。

PrintWriter(Writer out, boolean autoFlush) 为指定的字符输入流创建一个 PrintWriter,如果 autoFlush 为 true,则 println、printf 或 format 方法将刷新输出缓冲区。

PrintWriter(OutputStream out) 为指定的字节输入流创建一个 PrintWriter。

PrintWriter(OutputStream out, boolean autoFlush) 为指定的字节输入流创建一个 PrintWriter,如果 autoFlush 为 true,则 println、printf 或 format 方法将刷新输出缓冲区。

PrintWriter(String fileName) 为指定的文件路径创建一个 PrintWriter

PrintWriter(String fileName, String csn) 为指定的文件路径创建一个 PrintWriter,并且通过指定的字符集对字符进行编码。

PrintWriter(File file) 为指定的文件对象创建一个 PrintWriter。

PrintWriter(File file, String scn) 为指定的文件对象创建一个 PrintWriter,并且通过指定的字符集对字符进行编码。

PrintWriter 常用方法:

方法 返回值类型 说明

print(String s) void 将字符串写入文件中

println(String s) void 将字符串写入文件中,多打印一个换行

printf(String s) void 将字符串写入文件中,按照指定格式写文件中

System.out 为系统自带的 printStream,表示控制台的标准 Java 对象,通过它能够向控制台输入数据。

使用 PrintWriter 必须使用 close 方法关闭文件,或者使用 try-with-resources 自动关闭资源,狗则数据就不能正确地保存在文件中。

示例代码:

import java.io.File

import java.io.FileInputStream

import java.io.FileNotFoundException

import java.io.IOException

public class FileInputStreamTest1

{

public static void main(String[] args)

{

try

{

File file=new File("F:\\io\\abc.txt")

FileInputStream fis=new FileInputStream(file)

int first=fis.read()

System.out.print("文件中第一个字节是:"+first+"字符形式为:"+(char)first)

int b=0

while((b=fis.read())!=-1)

{

System.out.print((char)b+" ")

}

fis.close()

}

catch (FileNotFoundException e)

{

System.out.println("操作的文件不存在")

}

catch(IOException e)

{

            System.out.println("发生IO操作异常")

}

}

}