04、Netty 基础 之 NIO之Channel

一、基本介绍

1、 NIO的通道类似于流,但有些区别;

1、通道可以同时进行读写,而流只能读或者只能写
2、通道可以实现异步读写数据
3、通道可以从缓冲读数据,也可以写数据到缓冲

 

2、 BIO中的stream是单向的,例如FileInputStream对象只能进行读取数据的操作,而NIO中的通道(Channel)是双向的,可以读操作,也可以写操作;

3、 Channel在NIO中是一个接口;

public interface Channel extends Closeable

4、 常用的Channel类有:FileChannel、DatagramChannel、ServerSocketChannel(类似ServerSocket)、SocketChannel(类似Socket);

真实类型:

 

5、 FileChannel用于文件的数据读写,DatagramChannel用于UDP的数据读写,ServerSocketChannel和SocketChannel用于TCP的数据读写;

二、FileChannel类

FileChannel类主要用来对本地文件进行IO操作,常见方法有;

读和写是站在channel通道的角度

1、public abstract int read(ByteBuffer dst):从通道读取数据并放到缓冲区中
2、public abstract int write(ByteBuffer src):把缓冲区的数据写到通道中
3、public abstract long transferFrom(ReadableByteChannel src, long position, long count):从目标通道中复制数据到当前通道
4、public abstract long transferTo(long position, long count, WritableByteChannel target):把数据从当前通道复制给目标通道

三、案例1:本地文件写数据

1、 使用ByteBuffer(缓冲)和FileChannel(通道),将“hello,你好”写入到file01.txt中;
2、 文件不存在就创建;
3、 代码

![ ][nbsp2];

NIOFileChannel01.java

package netty.channel;

import java.io.FileOutputStream;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;

public class NIOFileChannel01 {
	public static void main(String[] args) throws Exception {
		String str = "hello,你好";
		//创建一个输出流->包装到channel中
		FileOutputStream fileOutputStream = new FileOutputStream("d:\\file01.txt");
		
		//通过fileOutputStream输出流获取对应的FileChannel
		//这个fileChannel真实类型是FileChannelImpl
		FileChannel fileChannel = fileOutputStream.getChannel();
		
		//创建一个缓冲区ByteBuffer
		ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
		
		//将str放入到byteBuffer中
		byteBuffer.put(str.getBytes());
		
		//对byteBuffer进行flip
		byteBuffer.flip();
		
		//将byteBuffer里的数据,写入到fileChannel
		fileChannel.write(byteBuffer);
		
		//关闭流
		fileOutputStream.close();
	}
}

读写翻转前:
 

翻转后:
 

四、案例2:本地文件读数据

1、 使用ByteBuffer(缓冲)和FileChannel(通道),将file01.txt中的数据读入到程序,并显示在控制台屏幕;
2、 假定文件已经存在;
3、 代码;
 

NIOFileChannel02.java

package netty.channel;

import java.io.File;
import java.io.FileInputStream;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;

public class NIOFileChannel02 {
	public static void main(String[] args) throws Exception {
		//创建文件的输入流
		File file = new File("d:\\file01.txt");
		FileInputStream fileInputStream = new FileInputStream(file);
		
		//通过fileInputStream获取对应的FileChannel -> 实际类型FileChannelImpl
		FileChannel fileChannel = fileInputStream.getChannel();
		
		//创建缓冲区
		ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
		
		//将通道的数据读入到byteBuffer中
		fileChannel.read(byteBuffer);
		
		//将byteBuffer的字节数据转成String
		System.out.println(new String(byteBuffer.array())); //返回buffer中的字节数组hb
		
		//关闭流
		fileInputStream.close();
	}
}

五、案例3:使用一个Buffer完成文件读取

1、 使用FileChannel(通道)和方法read、write,完成文件的拷贝;
2、 拷贝一个文本文件1.txt到2.txt,放在项目下即可;
3、 代码;
 

NIOFileChannel03.java

package netty.channel;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;

public class NIOFileChannel03 {
	public static void main(String[] args) throws Exception {
		//创建文件的输入流
		FileInputStream fileInputStream = new FileInputStream("d:\\file01.txt");
		//获取输入流对象的channel
		FileChannel fileChannel01 = fileInputStream.getChannel();
		
		//文件输出流对象
		FileOutputStream fileOutputStream = new FileOutputStream("d:\\file02.txt");
		//获取输入流对象的channel
		FileChannel fileChannel02 = fileOutputStream.getChannel();
		
		ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
		
		while(true) {
			//读之前有个重要操作,一定不要忘了
			byteBuffer.clear(); //复位:The position is set to zero, the limit is set to the capacity, and the mark is discarded
			
			//循环读取
			int read = fileChannel01.read(byteBuffer);
			System.out.println("read = " + read);
			if (read == -1) {
				//表示读完
				break;
			}
			
			//读写切换
			byteBuffer.flip();
			
			//将buffer中的数据写入到fileChannel02
			fileChannel02.write(byteBuffer);
		}
		
		//关闭相关的流
		fileInputStream.close();
		fileOutputStream.close();
	}
}

六、案例4:拷贝文件transferFrom方法

1、 使用FileChannel(通道)和方法transferFrom,完成文件的拷贝;
2、 拷贝一张图片;
3、 代码;
NIOFileChannel04.java

package netty.channel;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.nio.channels.FileChannel;

public class NIOFileChannel04 {
	public static void main(String[] args) throws Exception {
		//创建输入流
		FileInputStream fileInputStream = new FileInputStream("d:\\a.jpg");
		//创建输出流
		FileOutputStream fileOutputStream = new FileOutputStream("d:\\a2.jpg");
		
		//获取各个流对应的fileChannel
		FileChannel source = fileInputStream.getChannel();
		FileChannel dest = fileOutputStream.getChannel();
		
		//使用transferFrom完成拷贝
		dest.transferFrom(source, 0, source.size());
		
		//关闭通道和流
		source.close();
		dest.close();
		fileInputStream.close();
		fileOutputStream.close();
	}
}

七、ServerSocketChannel类

1、 ServerSocketChannel在服务端监听新的客户端Socket连接;

2、 相关方法;
public static ServerSocketChannel open() throws IOException:得到一个ServerSocketChannel通道
public final ServerSocketChannel bind(SocketAddress local) throws IOException:设置服务器端端口号
public final SelectableChannel configureBlocking(boolean block) throws IOException:设置阻塞或非阻塞模式,取值false表示采用非阻塞模式
public SocketChannel accept() throws IOException:接受一个连接,返回代表这个连接的通道对象
public final SelectionKey register(Selector sel, int ops) throws ClosedChannelException:注册一个选择器并设置监听事件

3、 ServerSocketChannel和SocketChannel;
ServerSocketChannel继承自AbstractSelectableChannel
 

SocketChannel也继承自AbstractSelectableChannel
 

但是SocketChannel实现的接口更多,它更重要的功能是对数据的读和写

八、SocketChannel类

1、 SocketChannel,网络IO通道,具体负责进行读写操作,NIO把缓冲区的数据写入通道,或者把通道里的数据读到缓冲区;

2、 相关方法;
public static SocketChannel open() throws IOException:得到一个SocketChannel通道
public final SelectableChannel configureBlocking(boolean block) throws IOException:设置阻塞或非阻塞模式,取值false表示采用非阻塞模式
public abstract boolean connect(SocketAddress remote) throws IOException:连接服务器
public abstract boolean finishConnect() throws IOException:如果上面的方法连接失败,接下来就要通过该方法完成连接操作
public abstract int write(ByteBuffer src) throws IOException:往通道里写数据
public abstract int read(ByteBuffer dst) throws IOException:从通道里读数据
public final SelectionKey register(Selector sel, int ops, Object att) throws ClosedChannelException:注册一个选择器并设置监听事件,最后一个参数可以设置共享数据
public final void close() throws IOException:关闭通道