JavaIO 面试题

1. Java字节流与字符流

83147a4a07cede3f8336bc2e8321d37f.png

2. 字节流和字符流哪个好?怎么选择?

大多数情况下使用字节流会更好,因为大多数时候 IO 操作都是直接操作磁盘文件,所以这些流在传输时都是以字节的方式进行的(图片等都是按字节存储的)

如果对于操作需要通过 IO 在内存中频繁处理字符串的情况使用字符流会好些,因为字符流具备缓冲区,提高了性能。

3. 什么是缓冲区?有什么作用?

缓冲区就是一段特殊的内存区域,很多情况下当程序需要频繁地操作一个资源(如文件或数据库)则性能会很低,所以为了提升性能就可以将一部分数据暂时读写到缓存区,以后直接从此区域中读写数据即可,这样就显著提升了性。

对于 Java 字符流的操作都是在缓冲区操作的,所以如果我们想在字符流操作中主动将缓冲区刷新到文件则可以使用 flush() 方法操作。

4. 字符流和字节流有什么区别?

字符流和字节流的使用非常相似,但是实际上字节流的操作不会经过缓冲区(内存)而是直接操作文本本身的,而字符流的操作会先经过缓冲区(内存)然后通过缓冲区再操作文件。

5. 什么是Java序列化,如何实现Java序列化?

序列化就是一种用来处理对象流的机制,将对象的内容进行流化。可以对流化后的对象进行读写操作,可以将流化后的对象传输于网络之间。序列化是为了解决在对象流读写操作时所引发的问题

序列化的实现:将需要被序列化的类实现Serialize接口,没有需要实现的方法,此接口只是为了标注对象可被序列化的,然后使用一个输出流(如:FileOutputStream)来构造一个ObjectOutputStream(对象流)对象,再使用ObjectOutputStream对象的write(Object obj)方法就可以将参数obj的对象写出

6. PrintStream、BufferedWriter、PrintWriter的比较?

PrintStream类的输出功能非常强大,通常如果需要输出文本内容,都应该将输出流包装成PrintStream后进行输出。它还提供其他两项功能。与其他输出流不同,PrintStream 永远不会抛出 IOException;而是,异常情况仅设置可通过 checkError 方法测试的内部标志。另外,为了自动刷新,可以创建一个 PrintStream

BufferedWriter:将文本写入字符输出流,缓冲各个字符从而提供单个字符,数组和字符串的高效写入。通过write()方法可以将获取到的字符输出,然后通过newLine()进行换行操作。BufferedWriter中的字符流必须通过调用flush方法才能将其刷出去。并且BufferedWriter只能对字符流进行操作。如果要对字节流操作,则使用BufferedInputStream

PrintWriter的println方法自动添加换行,不会抛异常,若关心异常,需要调用checkError方法看是否有异常发生,PrintWriter构造方法可指定参数,实现自动刷新缓存(autoflush)

7. BufferedReader属于哪种流,它主要是用来做什么的,它里面有那些经典的方法?

属于处理流中的缓冲流,可以将读取的内容存在内存里面,有readLine()方法,它,用来读取一行。

8. 什么是节点流,什么是处理流,它们各有什么用处,处理流的创建有什么特征?

节点流 直接与数据源相连,用于输入或者输出

处理流:在节点流的基础上对之进行加工,进行一些功能的扩展

处理流的构造器必须要 传入节点流的子类

9.流是怎么关闭的?

流一旦打开就必须关闭,使用close方法、放入finally语句块中(finally 语句一定会执行)、调用的处理流就关闭处理流、多个流互相调用只关闭最外层的流。

10. InputStream里的read()返回的是什么,read(byte[] data)是什么意思,返回的是什么值?

返回的是所读取的字节的int型(范围0-255)

read(byte [ ] data)将读取的字节储存在这个数组。返回的就是传入数组参数个数

11. OutputStream里面的write()是什么意思,write(byte b[], int off, int len)这个方法里面的三个参数分别是什么意思?

write将指定字节传入数据源

Byte b[ ]是byte数组

b[off]是传入的第一个字符、b[off+len-1]是传入的最后的一个字符 、len是实际长度

12. 说说BIO、NIO和AIO的区别


Java BIO: 同步并阻塞,服务器实现模式为一个连接一个线程,即客户端有连接请求时服务器端就需要启动一个线程进行处理,如果这个连接不做任何事情会造成不必要的线程开销,当然可以通过线程池机制改善。

Java NIO: 同步非阻塞,服务器实现模式为一个请求一个线程,即客户端发送的连接请求都会注册到多路复用器上,多路复用器轮询到连接有I/O请求时才启动一个线程进行处理。

Java AIO: 异步非阻塞,服务器实现模式为一个有效请求一个线程,客户端的I/O请求都是由OS先完成了再通知服务器应用去启动线程进行处理。

NIO比BIO的改善之处是把一些无效的连接挡在了启动线程之前,减少了这部分资源的浪费(因为我们都知道每创建一个线程,就要为这个线程分配一定的内存空间)

AIO比NIO的进一步改善之处是将一些暂时可能无效的请求挡在了启动线程之前,比如在NIO的处理方式中,当一个请求来的话,开启线程进行处理,但这个请求所需要的资源还没有就绪,此时必须等待后端的应用资源,这时线程就被阻塞了。

适用场景分析:

BIO方式适用于连接数目比较小且固定的架构,这种方式对服务器资源要求比较高,并发局限于应用中,JDK1.4以前的唯一选择,但程序直观简单易理解,如之前在Apache中使用。

NIO方式适用于连接数目多且连接比较短(轻操作)的架构,比如聊天服务器,并发局限于应用中,编程比较复杂,JDK1.4开始支持,如在 Nginx,Netty中使用。

AIO方式使用于连接数目多且连接比较长(重操作)的架构,比如相册服务器,充分调用OS参与并发操作,编程比较复杂,JDK7开始支持,在成长中,Netty曾经使用过,后来放弃。

以下是一些常见的 Java IO 面试题汇总: ### 基础概念类 1. **什么是 Java IO 流?** Java IO 流是用于在程序和外部资源(如文件、网络连接等)之间进行数据传输的抽象概念。流分为输入流和输出流,输入流用于从外部读取数据到程序中,输出流用于将程序中的数据写入到外部资源中。根据数据处理方式,又可分为字节流(以字节为单位处理数据,如 `InputStream` 和 `OutputStream`)和字符流(以字符为单位处理数据,如 `Reader` 和 `Writer`)。 2. **字节流和字符流的区别是什么?** 字节流以字节(8 位)为单位进行数据传输,可处理任意类型的数据,如图像、音频等;而字符流以字符(通常 16 位,取决于字符编码)为单位进行数据传输,主要用于处理文本数据。字节流在处理文本数据时,可能会出现乱码问题,因为它不了解字符编码;字符流则会根据指定的字符编码进行字符的编码和解码,能更好地处理文本。 3. **Java IO 流有哪些常见的类?** - 字节输入流:`FileInputStream`、`BufferedInputStream`、`DataInputStream` 等。 - 字节输出流:`FileOutputStream`、`BufferedOutputStream`、`DataOutputStream` 等。 - 字符输入流:`FileReader`、`BufferedReader`、`InputStreamReader` 等。 - 字符输出流:`FileWriter`、`BufferedWriter`、`OutputStreamWriter` 等。 ### 操作使用类 1. **如何使用 Java 读取文件内容?** 可以使用字节流或字符流来读取文件内容。以下是使用 `BufferedReader` 读取文件的示例代码: ```java import java.io.BufferedReader; import java.io.FileReader; import java.io.IOException; public class ReadFileExample { public static void main(String[] args) { try (BufferedReader br = new BufferedReader(new FileReader("test.txt"))) { String line; while ((line = br.readLine()) != null) { System.out.println(line); } } catch (IOException e) { e.printStackTrace(); } } } ``` 2. **如何使用 Java 写入文件内容?** 可以使用字节流或字符流来写入文件内容。以下是使用 `BufferedWriter` 写入文件的示例代码: ```java import java.io.BufferedWriter; import java.io.FileWriter; import java.io.IOException; public class WriteFileExample { public static void main(String[] args) { try (BufferedWriter bw = new BufferedWriter(new FileWriter("test.txt"))) { bw.write("Hello, World!"); } catch (IOException e) { e.printStackTrace(); } } } ``` 3. **简述 `BufferedInputStream` 和 `BufferedOutputStream` 的作用。** `BufferedInputStream` 和 `BufferedOutputStream` 是缓冲流,它们内部有一个缓冲区。在使用 `BufferedInputStream` 读取数据时,会先将数据从底层输入流读取到缓冲区中,后续的读取操作会先从缓冲区中获取数据,减少了与底层数据源的交互次数,提高了读取效率;`BufferedOutputStream` 则是在写入数据时,先将数据写入到缓冲区,当缓冲区满或者调用 `flush()` 方法时,才将缓冲区中的数据一次性写入到底层输出流,减少了与底层输出目标的交互次数,提高了写入效率。 ### 高级特性类 1. **什么是序列化和反序列化?如何实现 Java 对象的序列化和反序列化?** 序列化是指将 Java 对象转换为字节序列的过程,反序列化则是将字节序列恢复为 Java 对象的过程。实现序列化的类需要实现 `java.io.Serializable` 接口,该接口是一个标记接口,没有方法。以下是实现序列化和反序列化的示例代码: ```java import java.io.*; class Person implements Serializable { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public int getAge() { return age; } } public class SerializationExample { public static void main(String[] args) { // 序列化 try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("person.ser"))) { Person person = new Person("John", 30); oos.writeObject(person); } catch (IOException e) { e.printStackTrace(); } // 反序列化 try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("person.ser"))) { Person person = (Person) ois.readObject(); System.out.println("Name: " + person.getName() + ", Age: " + person.getAge()); } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); } } } ``` 2. **`RandomAccessFile` 有什么特点和用途?** `RandomAccessFile` 既可以作为输入流,也可以作为输出流,并且可以在文件的任意位置进行读写操作,即支持随机访问。它有一个文件指针,可以通过 `seek()` 方法将文件指针移动到指定的位置,然后进行读写操作。常用于需要随机访问文件内容的场景,如数据库文件的读写、文件的分块读取等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Thomas.Sir

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值