Java nio

1. java nio

java nio 意为java new IO. 它与java IO有以下三点区别:

  • nio是面向缓存的, 而io是面向的.
  • nio是非阻塞的, 而io是完全阻塞的.
  • 在nio中, 一个Selector可以管理多个通道, 而io则无此能力.

2. nio的三个关键组件?

  • Channel: 通道, 即数据源.
  • Buffer: 缓冲区
  • Selector: 选择器, 一个Selector可以管理多个Channel.

数据可以从Channel读入Buffer, 也可以从Buffer写入Channel.

3. Channel

Channel定义了以下几个接口:

  • Channel: 所有Channel的基类.
boolean isOpen();
  • ReadableByteChannel: 定义了将数据从Channel读入Buffer的操作,
// 返回读取的字节数, 读完则返回`-1`.
int read(Buffer b);
  • WritableByteChannel: 定义了将数据从Buffer写入Channel的操作.
// 返回写入的字节数,可能为零
int write(Buffer b);

注:只会写入未读过的数据.

  • GatheringByteChannel: 将多个Buffer中的数据依次写入Channel.
// 返回写入的字节数,可能为零
long write(Buffer[] fs);
long write(Buffer[] fs, int offset, int length);

注:只会写入未读过的数据.

  • ScatteringByteChannel: 将Channel中的数据依次读入多个Buffer中
long read(Buffer[] bs);
long read(Buffer[] bs, int offset, int length);
  • ByteChannel:
    继承了WritableByteChannelReadableByteChannel.没有定义新的方法.

  • InterruptibleChannel: 通道的异步关闭

void close();
3.1 FileChannel

操作文件
可以通过FileInputStream, FileOutputStream, RandomAccessFile对象的getChannel()方法获取.

// 通道中文件的位置, 返回从文件开始到当前位置的字节数
long position();
// 设置文件的位置
FileChannel position(long newPosition);
// 文件大小(字节)
long size();
// 将文件截取为指定大小, 返回截取后的文件大小(字节)
FileChannel truncate(long size);
// 将通道中文件的更新写入磁盘
void force(boolean metaData);
// 将此通道的文件写入到给定的可写入通道, 返回实际写入的字节数
long transferTo(long position, long count, WritableByteChannel channel);
// 从给定的通道读入数据, 返回实际读取的字节数
long transferFrom(ReadableByteChannel src, long position, long count);
// 将此通道的文件区域直接映射到内存中。
MappedByteBuffer map(FileChannel.MapMode mode, long position, long size);
// 获取锁, 会阻塞
FileLock lock(); 
// 获取锁, 会阻塞
FileLock lock(long position, long size, boolean shared);
// 获取锁, 不会阻塞
FileLock trylock();
// 获取锁, 不会阻塞
FileLock trylock(long position, long size, boolean shared);
3.2 DatagramChannel

处理UDP包的通道.

// 连接通道的套接字
DatagramChannel connect(SocketAddress remote);
// 断开连接
DatagramChannel disConnect();
// 是否连接
boolean isConnected();
// 打开通道
DatagramChannel open();
// 接收数据
SocketAddress receive(ByteBuffer dst);
// 发送数据
int send(ByteBuffer src, SocketAddress target);
// 获取此通道的套接字
DatagramSocket socket();
// 返回此通道所支持的操作
int validOps();
3.3 SocketChannel

处理TCP包数据

// 连接
boolean connect(SocketAddress remote);
// 是否完成连接
boolean finishConnect();
// 是否已连接
boolean isConnected();
// 是否正在进行连接操作
boolean isConnectionPending();
// 打开通道
SocketChannel open();
// 打开通道并连接到远程地址
SocketChannel open(SocketAddress remote);
// 获取套接字
Socket socket();
// 返回通道所支持的操作
int validOps();
3.4 ServerSocketChannel

监听套接字

// 接收连接并生成一个新的SocketChannel
SocketChannel accept();
// 打开通道
ServerSocketChannel open();
// 获取通道的服务器套接字
ServerSocket socket();
// 返回通道所支持的操作
int validOps();
3.5 SelectableChannel
// 设置通道的阻塞模式, 在将其注册到Selector时必须要设置其为false.
SelectableChannel configureBlocking(boolean block);
// 将通道注册到Selector
SelectionKey register(Selector  sel, int ops, Object att);

注:DatagramChannel, SocketChannel, ServerSocketChannel实现了该接口, FileChannel未实现该接口, 故FileChannel不能注册到Selector.

4. Buffer

缓冲区, 除了boolean以外所有的基本类型都一个与之对应的Buffer.

// Buffer的容量
int capacity();
// 清除Buffer, 并未清除缓冲区的实际内容, 只是将其当前位置置0, 限制设置为容量, 丢弃了标记.
Buffer clear();
// 首先将限制设置为当前位置,然后将位置设置为 0。如果已定义了标记,则丢弃该标记。
// 在read和put操作之后, 调用此方法后就可以write或get了.
Buffer flip();
// 是否有剩余
boolean hasRemaining();
// 是否是直接缓冲区
boolean isDirect();
// 是否只读
boolean isReadOnly();
// Buffer的限制
int limit();
// 设置Buffer的限制
Buffer limit(int newLimit);
// 在当前位置设置标记
Buffer mark();
// 当前位置
int position();
// 设置新的位置
Buffer position(int newPosition);
// 返回剩余
int remaining();
// 将当前位置更改为上次mark()的位置
Buffer reset();
// 可以从头开始重新读取Buffer中的数据
Buffer rewind();
4.1 ByteBuffer

字节缓冲区

// 分配给定容量的缓冲区
ByteBuffer allocate(int capacity);
ByteBuffer allocateDirect(int capacity);
// 压缩缓冲区, 删除已读数据, 未读数据前移, 然后接着往后写入.
ByteBuffer compact();
// 创建共享此缓冲区内容的新的字节缓冲区。
ByteBuffer duplicate();
// 创建共享此缓冲区内容的新的字节缓冲区
ByteBuffer slice();
// 创建此字节缓冲区的视图,作为 char 缓冲区。
// short, int, long, float, double都有与之对应的asXxxBuffer()
CharBuffer asCharBuffer();
// 缓冲区当前(或指定)位置的字节
byte get();
byte get(int index);
// 将此缓冲区的字节传输到给定的目标数组中
ByteBuffer get(byte[] dst);
// 读取此缓冲区的当前位置之后的两个字节,并将它们组成 char 值,然后将该位置增加 2。
// short, int, long, float, double都有与之对应的getXxx()方法
char getChar();
char getChar(int index);

// 将给定的字节写入缓冲区
ByteBuffer put(byte b);
ByteBuffer put(int index, byte b);
ByteBuffer put(byte[] src);
ByteBuffer put(ByteBuffer src);

// 将指定 char 值的字节按顺序写入到此缓冲区的当前位置,然后将该位置增加 2。
// short, int, long, float, double都有与之对应的putXxx()方法
ByteBuffer putChar(char value);
ByteBuffer putChar(int index, char value);

// 将 byte 数组包装到缓冲区中。缓冲区修改将导致数组修改,反之亦然。
ByteBuffer wrap(byte[] array);

ShortBuffer, CharBuffer, IntBuffer, LongBuffer, FloatBuffer, DoubleBufferByteBuffer类似.

5. Selector

一个Selector可以处理多个通道.

// 关闭选择器
void close();

boolean isOpen();
// 返回此选择器的键集。
Set<SelectionKey> keys();
Set<SelectionKey> selectKeys();
// 打开一个选择器
Selector open();
// 返回创建此通道的提供者。
SelectorProvider provider();
// 选择一组键,其相应的通道已为 I/O 操作准备就绪。会阻塞.  没有返回0
int select();
int select(long timeout);
// 选择一组键,其相应的通道已为 I/O 操作准备就绪。不会阻塞
int selectNow();
// 使尚未返回的第一个选择操作立即返回。
Selector wakeup();

6. SelectionKey

选择器Selector与Channel的对应关系.
常量:

  • OP_ACCEPT:
  • OP_CONNECT:
  • OP_READ:
  • OP_WRITE:

方法:

// 设置附加对象
Object attach(Object obj)
// 获取附加对象
Object attach();
// 取消此key表示的Channel与Selector之间的注册关系
void cancel();
// 获取Channel
SelectableChannel channel();
// 获取注册的操作集
int interestOps();
// 设置操作集
SelectionKey interestOps(int ops);
// 是否可接受新的连接
boolean isAcceptable();
// 是否已连接
boolean isConnectable();
// 是否可读
boolean isReadable();
// 是否可写
boolean isWritable();
// 是否有效
boolean isValid();
// 已就绪的操作集
readyOps();
// 获取Selector
Selector selector();

关于操作集位运算:
| : 表示相加.
&~: 表示有则取消
&: 结果为0表示没有, 大于0表示有

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容