FileChannel 文件流的简单使用

简介: FileChannel 文件流的简单使用

前言

在文章NIO 下的 ByteBuffer简单学习 中, 我们有写过一个简单的文件读取案例, 在案例中有使用到 Buffer 和 Channel, 关于 Buffer 的简单使用可以看下面两篇文章

关于 Channel 主要分为以下几种, 本篇文章是对 FileChannel 的讲解:

  • FileChannel: 文件通道, 主要用于对文件的读写
  • DatagramChannel: 数据包通道, 可以发送和接受 UPD 的数据包
  • SocketChannel: 套接字通过, 接收 TCP 数据包的读写
  • ServerSocketChannel: 服务器套接字通道, 监听新进来的 TCP 连接, 为每一个新连接都创建一个 SocketChannel

FileChannel

注意: FileChannel 只能工作在阻塞模式下
复制代码

新建

FileChannel 是一个抽象类, 所以不能直接创建对象

网络异常,图片无法展示
|

创建一个 FileChannel 有以下三种方式:

  • 创建一个 FileInputStream 对象, 但是该对象获取到的 Channel 只能读取
  • 创建一个 FileOutputStream 对象, 但是该对象获取到的 Channel 只能写入
  • 创建一个 RandomAccessFile 对象, 该对象能否读写是根据构造 RandomAccessFile 时设置的读写模式设定的

注意: Channel 使用之后必须关闭

不主动 close 并且 未触发gc , 那么 连接句柄 将被 一直占用, 如果此时使用的是连接池方式, 将造成连接池中的连接不能及时的被回收问题

在调用 FileInputStream 、 FileOutputStream 和 RandomAccessFile 的 close 方法会间接调用 Channel 的 close 方法

实现文件的读写

通过以下代码可以简单的通过 FileInputStream 和 FileOutputStream 来实现对文件的读写

网络异常,图片无法展示
|

但是这里有个问题, FileChannel 的 transferTo 只能传输 2G 以内的数据, 超过 2G 就传输不了了,

下面是 FileChannel 的 tarnsferTo 方法, 可以看到他是存在返回值的, 这个返回值就代表着还剩下多少字节的内容没有进行传输, 所以我们可以使用一个 for 循环来对当前的代码进行改进

网络异常,图片无法展示
|

改进后的代码如下所示

网络异常,图片无法展示
|

本篇文章所有代码

public static void main(String[] args) {
    try(
            FileChannel inputChannel = new FileInputStream("test1.txt").getChannel();
            FileChannel outputChannel = new FileOutputStream("test2.txt").getChannel()
    ){
        // 获取到 输入流 的大小
        long size = inputChannel.size();
        // res 代表剩余多少字节没有进行传输
        for (long res = size; res > 0; ){
            // 将 inputChannel 流读取到的内容通过 outputChannel 传输到指定的地址
            // 效率高, 底层会利用操作系统的 零拷贝 进行优化, 但是一次只能传输 2G 的数据
            // 该方法返回值为 剩余未传输的 字节数
            res = inputChannel.transferTo(0, inputChannel.size(), outputChannel);
        }
    }catch (Exception e){
        e.printStackTrace();
    }
}
复制代码



目录
相关文章
|
2月前
|
存储 Java
序列化流 ObjectInputStream 和 ObjectOutputStream 的基本使用【 File类+IO流知识回顾④】
这篇文章介绍了Java中ObjectInputStream和ObjectOutputStream类的基本使用,这两个类用于实现对象的序列化和反序列化。文章解释了序列化的概念、如何通过实现Serializable接口来实现序列化,以及如何使用transient关键字标记不需要序列化的属性。接着,通过示例代码演示了如何使用ObjectOutputStream进行对象的序列化和ObjectInputStream进行反序列化。
序列化流 ObjectInputStream 和 ObjectOutputStream 的基本使用【 File类+IO流知识回顾④】
|
2月前
|
Java
缓冲流和转换流的使用【 File类+IO流知识回顾③】
这篇文章介绍了Java中缓冲流(BufferedInputStream, BufferedOutputStream, BufferedReader, BufferedWriter)和转换流(InputStreamReader, OutputStreamWriter)的使用,包括它们的构造方法和如何利用它们提高IO操作的效率及处理字符编码问题。
缓冲流和转换流的使用【 File类+IO流知识回顾③】
|
2月前
|
安全 C# 开发者
C# 一分钟浅谈:文件操作与文件流详解
【9月更文挑战第4天】在日常开发中,文件的读写是基本而重要的任务。C# 通过 `System.IO` 命名空间提供了多种工具,如 `FileStream`、`StreamReader` 和 `StreamWriter` 等,用于处理文件和流。本文从基础概念入手,详细介绍了这些类的使用方法,并讨论了常见错误及其避免策略,包括文件不存在、权限问题和文件被占用等。通过示例代码,展示了如何创建、读取文件以及进行二进制数据操作,并强调了异常处理和性能优化的重要性。掌握这些技巧对于提升编程能力至关重要。
116 2
|
6月前
IO流:字符输入流Reader的超详细用法及底层原理
IO流:字符输入流Reader的超详细用法及底层原理
|
6月前
|
移动开发 Java Linux
IO流:字节输出流FileOutputStream的超详细用法
IO流:字节输出流FileOutputStream的超详细用法
|
存储 安全 编译器
C#中使用I/O文件流
流,即是二进制数值,文件和流 I/O(输入/输出)是指在存储媒介中传入或传出数据。在 .NET 中,System.IO命名空间包含允许以异步方式和同步方式对数据流和文件进行读取和写入操作的类型。这些命名空间还包含对文件执行压缩和解压缩的类型,以及通过管道和串行端口启用通信的类型。命名空间:System.IO程序集:System.Runtime.dll。
93 1
|
存储 缓存 Java
【Java I/O 流】文件字节输入输出流:FileInputStream & FileOutputStream(文件读写案例)
本篇文章主要讲述“文件字节输入流”和“文件字节输出流”两个类的使用,并列举了一些案例,帮助学习。
248 0
|
存储 Java 调度
文件:IO流
文件:IO流
104 0
文件:IO流