(三)NIO元件Channel+ByteBuffer操作檔案【玩轉Netty系列】

君哥聊程式設計發表於2020-11-09

FileChannel

Java NIO中的FileChannel是一個連線到檔案的通道。可以通過檔案通道讀寫檔案,我們可以通過InputStream、OutputStreamRandomAccessFile來獲取它的例項,FileChannel無法設定為非阻塞模式。

	FileOutputStream fos = new FileOutputStream("F:\\abcd.txt");
	FileChannel fosChannel = fos.getChannel();

常用API

  1. public int read(ByteBuffer dst):從通道讀取資料並放到快取區中
  2. public int write(ByteBuffer src):把緩衝區的資料寫到通道中
  3. public long transferFrom(ReadableByteChannel src,long position,long count):從目標通道中複製資料到當前通道
  4. public long transferTo(long position,long count,WritableByteChannel target):把資料從當前通道複製給目標通道

使用案例

以下案例中會使用到ByteBuffer,它的底層的本質是一個陣列,就是用來存放內容的,這裡你可以看成是一個byte[],在本系列的上一節有專門針對ByteBuffer進行講解。

  1. 使用FileChannel.write方法像磁碟寫入一個檔案
    public static void write() throws IOException {
        //通過輸出流獲取FileChannel
        FileOutputStream fos = new FileOutputStream("F:\\abc.txt");
        FileChannel channel = fos.getChannel();
    
    	//檔案中的內容
        String str = "hello,it235";
        //初始化空間,並將byte陣列放入buffer中
        ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
        ByteBuffer put = byteBuffer.put(str.getBytes());
    
        //翻轉buffer,將緩衝區的資料寫入通道
        put.flip();
        channel.write(put);
        fos.close();
    }
    
  2. 使用FileChannel.read讀取磁碟中的檔案
    public static void read() throws IOException {
        //IO流讀取File
        File file = new File("F:\\abc.txt");
        FileInputStream fis = new FileInputStream(file);
    
        //獲取Channel
        FileChannel channel = fis.getChannel();
    
        //檔案內容長度
        long length = file.length();
    
    	//例項化檔案內容長度大小的緩衝區,用來臨時存放讀取出來的內容
        ByteBuffer byteBuffer = ByteBuffer.allocate((int)length);
        channel.read(byteBuffer);
        //取出位元組陣列,並輸出
        System.out.println(new String(byteBuffer.array()));
        fis.close();
    }
    
  3. 使用FileChannel.write+read方法進行檔案的拷貝
    
    /**
     * 拷貝檔案
     * @throws IOException
     */
    public static void copy() throws IOException {
    
        //使用IO流讀取檔案,並拿到FileChannel
        File file = new File("F:\\abc.txt");
        FileInputStream fis = new FileInputStream(file);
        FileChannel fisChannel = fis.getChannel();
    
        //指定IO輸出檔案位置,並拿到FIleChannel
        FileOutputStream fos = new FileOutputStream("F:\\abcd.txt");
        FileChannel fosChannel = fos.getChannel();
    
        //宣告緩衝區大小
        ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
        //迴圈讀取並寫入
        while(true){
    
            //從fisChannel中取出內容到buffer中
            int read = fisChannel.read(byteBuffer);
            //如果沒有內容了,則跳出迴圈
            if(read == -1){
                break;
            }else{
                //因為使用的事buffer,所以需要翻轉
                byteBuffer.flip();
                //使用fosChannel將buffer寫入檔案
                fosChannel.write(byteBuffer);
            }
            //每次使用完byteBuffer後需要復位
            byteBuffer.clear();
        }
        fos.close();
        fis.close();
    }
    
  4. 使用FileChannel.transferFrom或transferTo進行檔案的拷貝
    
    public static void copyApi() throws IOException {
    
        //指定IO輸入、輸出檔案位置
        FileInputStream fis = new FileInputStream("F:\\abc.txt");
        FileOutputStream fos = new FileOutputStream("F:\\abcd.txt");
    
        //原始檔Channel
        FileChannel sourceChannel = fis.getChannel();
        //目標檔案CHannel
        FileChannel destChannel = fos.getChannel();
    
        //A:拷貝檔案:從目標通道中複製資料到當前通道
        destChannel.transferFrom(sourceChannel , 0 , sourceChannel.size());
    
        //B:拷貝檔案2:從原始檔通道中複製資料到目標通道
        //sourceChannel.transferTo(0 , sourceChannel.size() , destChannel );
    
        //關閉資源
        sourceChannel.close();
        destChannel.close();
        fos.close();
        fis.close();
    }
    

到此案例結束,接下來,我們來學習更多的內容…

相關文章