java

关注公众号 jb51net

关闭
首页 > 软件编程 > java > FileChannel使用

详解NIO中FileChannel文件流的简单使用

作者:宁轩

NIO中Channel分为了很多种,包括FileChannel,DatagramChannel,SocketChannel和ServerSocketChannel,其中FileChannel主要用于对文件的读写,本文主要和大家探讨了FileChannel文件流的简单使用,希望对大家有所帮助

前言

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

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

FileChannel

注意: FileChannel 只能工作在阻塞模式下

新建

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

public abstract class FileChannel 
    extends AbstractInterruptibleChannel
    implements SeekableByteChannel, GatheringByteChannel,ScatteringByteChannel
    {
        /**
         * Initializes a new instance of this class.
         * */
         protected FileChannel(){}

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

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

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

实现文件的读写

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

public static void main(String[] args) {
    try(
        FileChannel inputChannel = new FileInputStream( name: "test1.txt").getChannel();
        FileChannel outputChannel = new FileOutputStream( name: "test2.txt").getChannel()
        ){
            //  inputChannel 流读取到的内容通过 outputChannel 传输到指定的地址
            inputChannel.transferTo( position: 0,inputChannel.size(),outputChannel);
        }catch (Exception e){
           e.printstackTrace();
        }
}

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

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

改进后的代码如下所示

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

本篇文章所有代码

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();
    }
}

到此这篇关于详解NIO中FileChannel文件流的简单使用的文章就介绍到这了,更多相关FileChannel使用内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文