Java IO 和 NIO 的基本概念和 API

一、 Java IO (Blocking IO)

  1. 基本概念:

    • Java IO 是 Java 平台提供的用于进行输入和输出操作的 API。
    • Java IO 基于 流 (Stream) 的模型,数据像水流一样从一个地方流向另一个地方。
    • Java IO 主要是 阻塞式 I/O (Blocking I/O),即线程在执行 I/O 操作时会被阻塞,直到操作完成。
    • 传统IO指的是java.io包下的部分组件(File, InputStream, OutputStream, Reader, Writer)。
  2. IO 流的分类:

    • 按数据传输方向:

      • 输入流 (Input Stream): 用于从数据源读取数据(例如,从文件、网络连接、键盘等)。 以 InputStreamReader 作为基类。
      • 输出流 (Output Stream): 用于将数据写入到目标(例如,写入到文件、网络连接、控制台等)。 以 OutputStreamWriter 作为基类。
    • 按数据传输单位:

      • 字节流 (Byte Stream): 以字节 (8 bits) 为单位进行数据传输。 以 InputStreamOutputStream 作为基类。 适用于处理二进制数据(例如,图片、音频、视频等)。
      • 字符流 (Character Stream): 以字符 (16 bits) 为单位进行数据传输。 以 ReaderWriter 作为基类。 适用于处理文本数据。
  3. 核心类和接口:

    • InputStream (字节输入流):

      • FileInputStream: 从文件中读取字节。
      • ByteArrayInputStream: 从字节数组中读取字节。
      • ObjectInputStream: 从对象流中读取对象。
      • BufferedInputStream: 带缓冲的字节输入流,提高读取效率。
    • OutputStream (字节输出流):

      • FileOutputStream: 向文件中写入字节。
      • ByteArrayOutputStream: 向字节数组中写入字节。
      • ObjectOutputStream: 向对象流中写入对象。
      • BufferedOutputStream: 带缓冲的字节输出流,提高写入效率。
    • Reader (字符输入流):

      • FileReader: 从文件中读取字符。
      • CharArrayReader: 从字符数组中读取字符。
      • BufferedReader: 带缓冲的字符输入流,提高读取效率。
      • InputStreamReader: 将字节输入流转换为字符输入流(需要指定字符编码)。
    • Writer (字符输出流):

      • FileWriter: 向文件中写入字符。
      • CharArrayWriter: 向字符数组中写入字符。
      • BufferedWriter: 带缓冲的字符输出流,提高写入效率。
      • OutputStreamWriter: 将字节输出流转换为字符输出流(需要指定字符编码)。
    • File: 表示文件或目录的抽象表示。

  4. IO 操作流程 (以读取文件为例):

    1. 创建 File 对象: 指定要读取的文件路径。
    2. 创建 FileInputStream 对象:File 对象作为参数传递给 FileInputStream 的构造方法,创建一个 FileInputStream 对象。
    3. 创建 BufferedInputStream 对象 (可选):FileInputStream 对象作为参数传递给 BufferedInputStream 的构造方法,创建一个 BufferedInputStream 对象,提高读取效率。
    4. 读取数据: 使用 read() 方法从输入流中读取数据。
    5. 关闭流: 在完成读取操作后,务必关闭输入流,释放资源(先关闭 BufferedInputStream,再关闭 FileInputStream)。
  5. 代码示例 (读取文件内容):

    import java.io.BufferedInputStream;
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.IOException;
    
    public class IOExample {
        public static void main(String[] args) {
            File file = new File("test.txt"); // 替换为你的文件路径
    
            try (FileInputStream fis = new FileInputStream(file);
                 BufferedInputStream bis = new BufferedInputStream(fis)) { // 使用 try-with-resources 语句,自动关闭流
    
                byte[] buffer = new byte[1024];
                int bytesRead;
    
                while ((bytesRead = bis.read(buffer)) != -1) {
                    // 处理读取到的数据
                    String data = new String(buffer, 0, bytesRead);
                    System.out.print(data);
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    

二、 Java NIO (Non-blocking IO)

  1. 基本概念:

    • Java NIO 是 Java 1.4 引入的一组新的 I/O API,旨在提供高性能、非阻塞的 I/O 操作。
    • NIO 使用 通道 (Channel)缓冲区 (Buffer) 的模型,而不是流。
    • NIO 主要是 非阻塞式 I/O (Non-blocking I/O),即线程在执行 I/O 操作时不会被阻塞,而是可以执行其他任务。
    • NIO 使用 选择器 (Selector) 来监听多个通道的事件,实现单线程管理多个连接。
  2. 核心组件:

    • 通道 (Channel):

      • 通道类似于流,但可以进行双向数据传输(既可以读取数据,也可以写入数据)。
      • 常见的通道:
        • FileChannel: 用于文件 I/O。
        • SocketChannel: 用于 TCP 网络 I/O (客户端)。
        • ServerSocketChannel: 用于 TCP 网络 I/O (服务器端)。
        • DatagramChannel: 用于 UDP 网络 I/O。
    • 缓冲区 (Buffer):

      • 缓冲区是用于存储数据的容器,本质上是一个字节数组 (ByteBuffer) 或字符数组 (CharBuffer)。
      • NIO 使用缓冲区进行数据传输,而不是直接从通道读取数据或向通道写入数据。
      • 常见的缓冲区:
        • ByteBuffer: 字节缓冲区。
        • CharBuffer: 字符缓冲区。
        • ShortBuffer: 短整型缓冲区。
        • IntBuffer: 整型缓冲区。
        • LongBuffer: 长整型缓冲区。
        • FloatBuffer: 浮点型缓冲区。
        • DoubleBuffer: 双精度浮点型缓冲区。
    • 选择器 (Selector):

      • 选择器允许单个线程监听多个通道的事件(例如连接建立、数据可读、数据可写等)。
      • 使用选择器可以避免为每个连接创建一个线程,从而提高并发性能。
  3. NIO 操作流程 (以读取 SocketChannel 数据为例):

    1. 创建 ServerSocketChannel 监听客户端连接。
    2. 创建 SocketChannel 接受客户端连接。
    3. SocketChannel 注册到 Selector 指定要监听的事件(例如 OP_READ, OP_WRITE, OP_CONNECT, OP_ACCEPT)。
    4. 创建 ByteBuffer 用于存储读取到的数据。
    5. 调用 selector.select() 方法: 阻塞等待有事件发生的通道。
    6. 获取就绪的通道: selector.selectedKeys() 返回所有就绪通道的集合。
    7. 处理事件: 遍历就绪通道的集合,根据不同的事件类型执行相应的操作(例如读取数据、写入数据)。
    8. 读取数据: 调用 channel.read(buffer) 从通道读取数据到缓冲区。
    9. 处理缓冲区数据: 从缓冲区读取数据并进行处理。
    10. 关闭通道和选择器: 在完成操作后,务必关闭通道和选择器,释放资源。
  4. 代码示例 (使用 SocketChannel 读取数据):

    import java.io.IOException;
    import java.net.InetSocketAddress;
    import java.nio.ByteBuffer;
    import java.nio.channels.SocketChannel;
    
    public class NIOExample {
        public static void main(String[] args) throws IOException {
            // 1. 创建 SocketChannel
            SocketChannel socketChannel = SocketChannel.open();
            socketChannel.connect(new InetSocketAddress("www.example.com", 80));
            socketChannel.configureBlocking(false); // 设置为非阻塞模式
    
            // 2. 创建 ByteBuffer
            ByteBuffer buffer = ByteBuffer.allocate(1024);
    
            // 3. 从 Channel 读取数据到 Buffer
            int bytesRead = socketChannel.read(buffer);
    
            while (bytesRead > 0) {
                // 切换到读模式
                buffer.flip();
    
                // 4. 从 Buffer 读取数据
                while (buffer.hasRemaining()) {
                    System.out.print((char) buffer.get());
                }
    
                // 清空 Buffer,准备下一次读取
                buffer.clear();
                bytesRead = socketChannel.read(buffer);
            }
    
            socketChannel.close();
        }
    }
    

三、 Java IO 与 NIO 的区别

特性Java IO (Blocking IO)Java NIO (Non-blocking IO)
数据传输方式基于流 (Stream)基于通道 (Channel) 和缓冲区 (Buffer)
I/O 模型阻塞式 I/O (Blocking I/O)非阻塞式 I/O (Non-blocking I/O)
选择器没有有 (Selector)
API简单易用相对复杂,需要理解通道、缓冲区、选择器等概念
性能性能较低 (高并发下)性能较高 (高并发下)
线程模型通常使用多线程模型 (每个连接一个线程)通常使用单线程多路复用模型 (一个线程管理多个连接)
适用场景低并发、连接数较少的应用,或者可以接受阻塞的场景高并发、连接数较多的应用,需要高性能和非阻塞的场景,例如网络服务器、聊天服务器等

四、 选择哪种 I/O 模式?

  • 如果你的应用是低并发、连接数较少,并且可以接受阻塞,那么 Java IO 仍然是一个不错的选择,因为它简单易用。
  • 如果你的应用是高并发、连接数较多,并且对性能要求很高,那么应该使用 Java NIO。
  • 一般情况建议直接使用NIO模型,性能更好。

总结

Java IO 和 NIO 都是 Java 平台提供的用于进行输入和输出操作的 API。 Java IO 基于流的模型,使用简单但性能较低; Java NIO 基于通道和缓冲区的模型,提供高性能、非阻塞的 I/O 操作。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/977868.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

vscode如何使用鼠标滚轮调整字体大小

1.打开设置 2.搜索Font Ligatures 3.编辑配置文件 4.修改代码并保存 修改前 修改后 在最后一行添加:“editor.mouseWheelZoom”: true 记得在上一行最后,加上英文版的“,”逗号 5.配置成功,再次按Ctrl鼠标滚轮便可以缩放了。

视频裂变加群推广分享引流源码

源码介绍 视频裂变加群推广分享引流源码 最近网上很火,很多人都在用,适合引流裂变推广 测试环境:PHP7.4(PHP版本不限制) 第一次访问送五次观看次数,用户达到观看次数后需要分享给好友或者群,好友必须点击推广链接后才会增加观看次…

redis小记

redis小记 下载redis sudo apt-get install redis-server redis基本命令 ubuntu16下的redis没有protected-mode属性,就算sudo启动,也不能往/var/spool/cron/crontabs写计划任务,感觉很安全 #连接到redis redis-cli -h 127.0.0.1 -p 6379 …

IDEA关闭SpringBoot程序后仍然占用端口的排查与解决

IDEA关闭SpringBoot程序后仍然占用端口的排查与解决 问题描述 在使用 IntelliJ IDEA 开发 Spring Boot 应用时,有时即使关闭了应用,程序仍然占用端口(例如:4001 端口)。这会导致重新启动应用时出现端口被占用的错误&a…

Vue04

自定义指令 directives是Vue的一个配置项 这里写自定义指令 自定义指令被调用的时机 指令与元素成功绑定时 指令所在的模板被重新解析时 函数式 <span v-big"n"></span> directives:{ big(element,binding){ element.innerText bingin…

岳阳市美术馆预约平台(小程序论文源码调试讲解)

第4章 系统设计 一个成功设计的系统在内容上必定是丰富的&#xff0c;在系统外观或系统功能上必定是对用户友好的。所以为了提升系统的价值&#xff0c;吸引更多的访问者访问系统&#xff0c;以及让来访用户可以花费更多时间停留在系统上&#xff0c;则表明该系统设计得比较专…

Linux 基本开发工具的使用(yum、vim、gcc、g++、gdb、make/makefile)

文章目录 Linux 软件包管理器 - yum理解什么是软件包和yum如何查看/查找软件包如何安装软件如何实现本地机器和云服务器之间的文件互传如何卸载软件 Linux 编辑器 - vim 的使用vim 的基本概念vim 的基本操作vim 命令模式各命令汇总vim 底行模式各命令汇总vim 的简单配置 Linux …

4部署kibana:5601

kibana 是一个基于浏览器页面的Elasticsearch前端展示工具&#xff0c;, 是一个开源和免费的工具 Kibana可以为 Logstash 和 ElasticSearch 提供的日志分析友好的 Web 界面, 可以帮你汇总、分析和搜索重要数据日志 1.安装-所有的es节点 # tar xf kibana-6.4.1-linux-x86_64.t…

1.介绍一下TCP/IP模型和OSI模型的区别【中高频】

OSI模型 将 这个协议 划分为7个不同的层级&#xff0c;分别为物理层、数据链路层、网络层、传输层、会话层、表示层和应用层&#xff0c;而TCP/IP模型只有4个层级&#xff0c;分别为网络接口层、网络层、传输层和应用层&#xff0c;其中应用层在用户态&#xff0c;传输层及以下…

【反爬】拦截comBusiness.js disable-devtool.js

一、现象 无法使用ctrls保存网页&#xff0c;但是可以在设置菜单中可以保存&#xff1b;无法使用F12和ctrlshifti打开开发者窗口&#xff0c;但是可以在设置菜单中打开&#xff1b;打开开发者窗口后网站快速关闭&#xff0c;说明被检测到了&#xff1b; 二、涉及js 打开设置菜…

【11】子网

区块链子网概述 什么是子网&#xff1f; 子网是在较大网络上下文中运行的较小网络&#xff0c;因此由对应的“主网”&#xff0c;主网即包含多个子网的较大网络或具有隶属关系的上一级网络。子网允许在主网中进行一些独立的事务或控制网络参数。 对于互联网而言&#xff0c;…

爬虫基础入门之爬取豆瓣电影Top250-Re正则的使用

网址:豆瓣电影 Top 250 本案例所需要的模块 requests (用于发送HTTP请求)re (用于字符串匹配和操作) 确定需要爬取的数据 &#xff1a; 电影的名称电影的年份电影的评分电影评论人数 一. 发送请求 模拟浏览器向服务器发送请求 准备工作 -分析页面: F12 or 右击点击检查 查看…

论文笔记(七十二)Reward Centering(五)

Reward Centering&#xff08;五&#xff09; 文章概括摘要附录B 理论细节C 实验细节D 相关方法的联系 文章概括 引用&#xff1a; article{naik2024reward,title{Reward Centering},author{Naik, Abhishek and Wan, Yi and Tomar, Manan and Sutton, Richard S},journal{arX…

Flash-03

1-问题&#xff1a;Flash软件画两个图形&#xff0c;若有部分重合则变为一个整体 解决方法1&#xff1a;两个图形分属于不同的图层 解决方法2&#xff1a;将每个图形都转化为【元件】 问题2&#xff1a;元件是什么&#xff1f; 在 Adobe Flash&#xff08;现在称为 Adobe Anim…

ssh配置 远程控制 远程协作 github本地配置

0.设备版本 windows11 ubuntu24.0.4 1.1 在 Linux 上启用 SSH 服务 首先&#xff0c;确保 Linux 计算机上安装并启用了 SSH 服务。 安装和启动 OpenSSH 服务&#xff08;如果未安装&#xff09; # 在终端安装 OpenSSH 服务&#xff08;如果尚未安装&#xff09; sudo apt …

C语言数据结构—堆的应用及Topk问题

目录 1、堆排序 1、把数组先原地调整成堆 1.1 向上调整 1.2 向下调整 1.3 两种调整方式的时间复杂度分析 2、进行排序 1、堆排序 堆排序即利用堆的思想来进行排序&#xff0c;总共分为两个步骤&#xff1a; 1、建堆 升序&#xff1a;建大堆 降序&#xff1a;建小堆 2、利…

贪心算法精品题

1.找钱问题 本题的贪心策略在于我们希望就可能的保留作用大的5元 class Solution { public:bool lemonadeChange(vector<int>& bills) {std::map<int ,int> _map;for(auto ch:bills){if(ch 5) _map[ch];else if(ch 10){if(_map[5] 0) return false;else{_m…

【Blender】三、材质篇--01,Blender材质基础 原理化BSDF

0 00:00:05,460 --> 00:00:09,980 好 材质篇上一张呢 我们做了12个模型 我知道大家很想把它晒出来 1 00:00:10,440 --> 00:00:17,360 但是咱们先把材质学了吧 学材质 我们只要抓住那些对精髓的东西就好了 能够用手试出来的东西呢 你 2 00:00:17,530 --> 00:00:30,37…

博客系统完整开发流程

前言 通过前⾯课程的学习, 我们掌握了Spring框架和MyBatis的基本使用, 并完成了图书管理系统的常规功能开发, 接下来我们系统的从0到1完成⼀个项⽬的开发. 企业开发的流程 1. 需求评审(产品经理(PM)会和运营(想口号),UI,测试,开发等沟通) ,会涉及到背景/目标/怎么做,可能会有多…

iOS App的启动与优化

App的启动流程 App启动分为冷启动和热启动 冷启动&#xff1a;从0开始启动App热启动&#xff1a;App已经在内存中&#xff0c;但是后台还挂着&#xff0c;再次点击图标启动App。 一般对App启动的优化都是针对冷启动。 App冷启动可分为三个阶段&#xff1a; dyld&#xff1a…