Java IO.md 21.2 KB
Newer Older
C
CyC2018 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
<!-- GFM-TOC -->
* [一、概览](#一概览)
* [二、磁盘操作](#二磁盘操作)
* [三、字节操作](#三字节操作)
    * [实现文件复制](#实现文件复制)
    * [装饰者模式](#装饰者模式)
* [四、字符操作](#四字符操作)
    * [编码与解码](#编码与解码)
    * [String 的编码方式](#string-的编码方式)
    * [Reader 与 Writer](#reader-与-writer)
    * [实现逐行输出文本文件的内容](#实现逐行输出文本文件的内容)
* [五、对象操作](#五对象操作)
    * [序列化](#序列化)
    * [Serializable](#serializable)
    * [transient](#transient)
* [六、网络操作](#六网络操作)
    * [InetAddress](#inetaddress)
    * [URL](#url)
    * [Sockets](#sockets)
    * [Datagram](#datagram)
* [七、NIO](#七nio)
    * [流与块](#流与块)
    * [通道与缓冲区](#通道与缓冲区)
    * [缓冲区状态变量](#缓冲区状态变量)
    * [文件 NIO 实例](#文件-nio-实例)
    * [选择器](#选择器)
    * [套接字 NIO 实例](#套接字-nio-实例)
    * [内存映射文件](#内存映射文件)
    * [对比](#对比)
* [八、参考资料](#八参考资料)
<!-- GFM-TOC -->


# 一、概览

Java 的 I/O 大概可以分成以下几类:

- 磁盘操作:File
- 字节操作:InputStream 和 OutputStream
- 字符操作:Reader 和 Writer
- 对象操作:Serializable
- 网络操作:Socket
- 新的输入/输出:NIO

# 二、磁盘操作

File 类可以用于表示文件和目录的信息,但是它不表示文件的内容。
C
CyC2018 已提交
48

C
CyC2018 已提交
49
递归地列出一个目录下所有文件:
C
CyC2018 已提交
50 51

```java
C
CyC2018 已提交
52 53 54 55 56 57 58 59 60 61 62
public static void listAllFiles(File dir) {
    if (dir == null || !dir.exists()) {
        return;
    }
    if (dir.isFile()) {
        System.out.println(dir.getName());
        return;
    }
    for (File file : dir.listFiles()) {
        listAllFiles(file);
    }
C
CyC2018 已提交
63 64 65
}
```

C
CyC2018 已提交
66
从 Java7 开始,可以使用 Paths 和 Files 代替 File。
郑永川 已提交
67

C
CyC2018 已提交
68
# 三、字节操作
C
CyC2018 已提交
69

C
CyC2018 已提交
70
## 实现文件复制
C
CyC2018 已提交
71 72

```java
C
CyC2018 已提交
73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88
public static void copyFile(String src, String dist) throws IOException {
    FileInputStream in = new FileInputStream(src);
    FileOutputStream out = new FileOutputStream(dist);

    byte[] buffer = new byte[20 * 1024];
    int cnt;

    // read() 最多读取 buffer.length 个字节
    // 返回的是实际读取的个数
    // 返回 -1 的时候表示读到 eof,即文件尾
    while ((cnt = in.read(buffer, 0, buffer.length)) != -1) {
        out.write(buffer, 0, cnt);
    }

    in.close();
    out.close();
C
CyC2018 已提交
89 90 91
}
```

C
CyC2018 已提交
92
## 装饰者模式
C
CyC2018 已提交
93

C
CyC2018 已提交
94
Java I/O 使用了装饰者模式来实现。以 InputStream 为例,
C
CyC2018 已提交
95

C
CyC2018 已提交
96 97 98
- InputStream 是抽象组件;
- FileInputStream 是 InputStream 的子类,属于具体组件,提供了字节流的输入操作;
- FilterInputStream 属于抽象装饰者,装饰者用于装饰组件,为组件提供额外的功能。例如 BufferedInputStream 为 FileInputStream 提供缓存的功能。
C
CyC2018 已提交
99

C
CyC2018 已提交
100
<div align="center"> <img src="pics/c2c2b633-c03a-426e-b436-5719a194667b.png"/> </div><br>
C
CyC2018 已提交
101

C
CyC2018 已提交
102
实例化一个具有缓存功能的字节流对象时,只需要在 FileInputStream 对象上再套一层 BufferedInputStream 对象即可。
C
CyC2018 已提交
103 104

```java
C
CyC2018 已提交
105 106
FileInputStream fileInputStream = new FileInputStream(filePath);
BufferedInputStream bufferedInputStream = new BufferedInputStream(fileInputStream);
C
CyC2018 已提交
107 108
```

C
CyC2018 已提交
109
DataInputStream 装饰者提供了对更多数据类型进行输入的操作,比如 int、double 等基本类型。
C
CyC2018 已提交
110

C
CyC2018 已提交
111
# 四、字符操作
C
CyC2018 已提交
112

C
CyC2018 已提交
113
## 编码与解码
C
CyC2018 已提交
114

C
CyC2018 已提交
115 116
编码就是把字符转换为字节,而解码是把字节重新组合成字符。

C
CyC2018 已提交
117 118
如果编码和解码过程使用不同的编码方式那么就出现了乱码。

C
CyC2018 已提交
119 120 121
- GBK 编码中,中文字符占 2 个字节,英文字符占 1 个字节;
- UTF-8 编码中,中文字符占 3 个字节,英文字符占 1 个字节;
- UTF-16be 编码中,中文字符和英文字符都占 2 个字节。
C
CyC2018 已提交
122

C
CyC2018 已提交
123
UTF-16be 中的 be 指的是 Big Endian,也就是大端。相应地也有 UTF-16le,le 指的是 Little Endian,也就是小端。
C
CyC2018 已提交
124

C
CyC2018 已提交
125
Java 的内存编码使用双字节编码 UTF-16be,这不是指 Java 只支持这一种编码方式,而是说 char 这种类型使用 UTF-16be 进行编码。char 类型占 16 位,也就是两个字节,Java 使用这种双字节编码是为了让一个中文或者一个英文都能使用一个 char 来存储。
C
CyC2018 已提交
126

C
CyC2018 已提交
127
## String 的编码方式
C
CyC2018 已提交
128

C
CyC2018 已提交
129
String 可以看成一个字符序列,可以指定一个编码方式将它编码为字节序列,也可以指定一个编码方式将一个字节序列解码为 String。
C
CyC2018 已提交
130 131

```java
C
CyC2018 已提交
132 133 134
String str1 = "中文";
byte[] bytes = str1.getBytes("UTF-8");
String str2 = new String(bytes, "UTF-8");
C
CyC2018 已提交
135 136 137
System.out.println(str2);
```

C
CyC2018 已提交
138
在调用无参数 getBytes() 方法时,默认的编码方式不是 UTF-16be。双字节编码的好处是可以使用一个 char 存储中文和英文,而将 String 转为 bytes[] 字节数组就不再需要这个好处,因此也就不再需要双字节编码。getBytes() 的默认编码方式与平台有关,一般为 UTF-8。
C
CyC2018 已提交
139 140

```java
C
CyC2018 已提交
141
byte[] bytes = str1.getBytes();
C
CyC2018 已提交
142
```
C
CyC2018 已提交
143

C
CyC2018 已提交
144
## Reader 与 Writer
C
CyC2018 已提交
145 146 147

不管是磁盘还是网络传输,最小的存储单元都是字节,而不是字符。但是在程序中操作的通常是字符形式的数据,因此需要提供对字符进行操作的方法。

C
CyC2018 已提交
148 149
- InputStreamReader 实现从字节流解码成字符流;
- OutputStreamWriter 实现字符流编码成为字节流。
C
CyC2018 已提交
150

C
CyC2018 已提交
151
## 实现逐行输出文本文件的内容
C
CyC2018 已提交
152 153

```java
C
CyC2018 已提交
154
public static void readFileContent(String filePath) throws IOException {
C
CyC2018 已提交
155

C
CyC2018 已提交
156 157
    FileReader fileReader = new FileReader(filePath);
    BufferedReader bufferedReader = new BufferedReader(fileReader);
C
CyC2018 已提交
158

C
CyC2018 已提交
159 160 161 162
    String line;
    while ((line = bufferedReader.readLine()) != null) {
        System.out.println(line);
    }
C
CyC2018 已提交
163

C
CyC2018 已提交
164 165 166 167
    // 装饰者模式使得 BufferedReader 组合了一个 Reader 对象
    // 在调用 BufferedReader 的 close() 方法时会去调用 Reader 的 close() 方法
    // 因此只要一个 close() 调用即可
    bufferedReader.close();
C
CyC2018 已提交
168 169 170
}
```

C
CyC2018 已提交
171
# 五、对象操作
C
CyC2018 已提交
172

C
CyC2018 已提交
173
## 序列化
C
CyC2018 已提交
174

C
CyC2018 已提交
175 176
序列化就是将一个对象转换成字节序列,方便存储和传输。

C
CyC2018 已提交
177 178
- 序列化:ObjectOutputStream.writeObject()
- 反序列化:ObjectInputStream.readObject()
C
CyC2018 已提交
179

C
CyC2018 已提交
180 181
不会对静态变量进行序列化,因为序列化只是保存对象的状态,静态变量属于类的状态。

C
CyC2018 已提交
182
## Serializable
C
CyC2018 已提交
183

C
CyC2018 已提交
184
序列化的类需要实现 Serializable 接口,它只是一个标准,没有任何方法需要实现,但是如果不去实现它的话而进行序列化,会抛出异常。
C
CyC2018 已提交
185

C
CyC2018 已提交
186
```java
C
CyC2018 已提交
187
public static void main(String[] args) throws IOException, ClassNotFoundException {
C
CyC2018 已提交
188

C
CyC2018 已提交
189 190
    A a1 = new A(123, "abc");
    String objectFile = "file/a1";
C
CyC2018 已提交
191

C
CyC2018 已提交
192 193 194
    ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream(objectFile));
    objectOutputStream.writeObject(a1);
    objectOutputStream.close();
C
CyC2018 已提交
195

C
CyC2018 已提交
196 197 198 199
    ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(objectFile));
    A a2 = (A) objectInputStream.readObject();
    objectInputStream.close();
    System.out.println(a2);
C
CyC2018 已提交
200 201
}

C
CyC2018 已提交
202
private static class A implements Serializable {
C
CyC2018 已提交
203

C
CyC2018 已提交
204 205
    private int x;
    private String y;
C
CyC2018 已提交
206

C
CyC2018 已提交
207 208 209 210
    A(int x, String y) {
        this.x = x;
        this.y = y;
    }
C
CyC2018 已提交
211

C
CyC2018 已提交
212 213 214 215
    @Override
    public String toString() {
        return "x = " + x + "  " + "y = " + y;
    }
C
CyC2018 已提交
216 217
}
```
C
CyC2018 已提交
218

C
CyC2018 已提交
219
## transient
C
CyC2018 已提交
220

C
CyC2018 已提交
221
transient 关键字可以使一些属性不会被序列化。
C
CyC2018 已提交
222

C
CyC2018 已提交
223
ArrayList 中存储数据的数组 elementData 是用 transient 修饰的,因为这个数组是动态扩展的,并不是所有的空间都被使用,因此就不需要所有的内容都被序列化。通过重写序列化和反序列化方法,使得可以只序列化数组中有内容的那部分数据。
C
CyC2018 已提交
224 225

```java
C
CyC2018 已提交
226
private transient Object[] elementData;
C
CyC2018 已提交
227 228
```

C
CyC2018 已提交
229
# 六、网络操作
C
CyC2018 已提交
230

C
CyC2018 已提交
231
Java 中的网络支持:
C
CyC2018 已提交
232

C
CyC2018 已提交
233 234 235 236
- InetAddress:用于表示网络上的硬件资源,即 IP 地址;
- URL:统一资源定位符;
- Sockets:使用 TCP 协议实现网络通信;
- Datagram:使用 UDP 协议实现网络通信。
C
CyC2018 已提交
237

C
CyC2018 已提交
238
## InetAddress
C
CyC2018 已提交
239

C
CyC2018 已提交
240
没有公有的构造函数,只能通过静态方法来创建实例。
C
CyC2018 已提交
241 242

```java
C
CyC2018 已提交
243 244
InetAddress.getByName(String host);
InetAddress.getByAddress(byte[] address);
C
CyC2018 已提交
245 246
```

C
CyC2018 已提交
247
## URL
C
CyC2018 已提交
248

C
CyC2018 已提交
249
可以直接从 URL 中读取字节流数据。
C
CyC2018 已提交
250 251

```java
C
CyC2018 已提交
252
public static void main(String[] args) throws IOException {
C
CyC2018 已提交
253

C
CyC2018 已提交
254
    URL url = new URL("http://www.baidu.com");
C
CyC2018 已提交
255

C
CyC2018 已提交
256 257
    /* 字节流 */
    InputStream is = url.openStream();
C
CyC2018 已提交
258

C
CyC2018 已提交
259 260
    /* 字符流 */
    InputStreamReader isr = new InputStreamReader(is, "utf-8");
C
CyC2018 已提交
261

C
CyC2018 已提交
262 263
    /* 提供缓存功能 */
    BufferedReader br = new BufferedReader(isr);
C
CyC2018 已提交
264

C
CyC2018 已提交
265 266 267 268
    String line;
    while ((line = br.readLine()) != null) {
        System.out.println(line);
    }
C
CyC2018 已提交
269

C
CyC2018 已提交
270
    br.close();
C
CyC2018 已提交
271 272 273
}
```

C
CyC2018 已提交
274
## Sockets
C
CyC2018 已提交
275

C
CyC2018 已提交
276 277 278
- ServerSocket:服务器端类
- Socket:客户端类
- 服务器和客户端通过 InputStream 和 OutputStream 进行输入输出。
C
CyC2018 已提交
279

C
CyC2018 已提交
280
<div align="center"> <img src="pics/f77f06b6-7359-4066-b85d-3f6c09ddf425.jpg"/> </div><br>
C
CyC2018 已提交
281

C
CyC2018 已提交
282
## Datagram
C
CyC2018 已提交
283

C
CyC2018 已提交
284 285
- DatagramSocket:通信类
- DatagramPacket:数据包类
C
CyC2018 已提交
286

C
CyC2018 已提交
287
# 七、NIO
C
CyC2018 已提交
288

C
CyC2018 已提交
289
新的输入/输出 (NIO) 库是在 JDK 1.4 中引入的,弥补了原来的 I/O 的不足,提供了高速的、面向块的 I/O。
C
CyC2018 已提交
290

C
CyC2018 已提交
291
## 流与块
C
CyC2018 已提交
292

C
CyC2018 已提交
293
I/O 与 NIO 最重要的区别是数据打包和传输的方式,I/O 以流的方式处理数据,而 NIO 以块的方式处理数据。
C
CyC2018 已提交
294

C
CyC2018 已提交
295
面向流的 I/O 一次处理一个字节数据:一个输入流产生一个字节数据,一个输出流消费一个字节数据。为流式数据创建过滤器非常容易,链接几个过滤器,以便每个过滤器只负责复杂处理机制的一部分。不利的一面是,面向流的 I/O 通常相当慢。
C
CyC2018 已提交
296

C
CyC2018 已提交
297
面向块的 I/O 一次处理一个数据块,按块处理数据比按流处理数据要快得多。但是面向块的 I/O 缺少一些面向流的 I/O 所具有的优雅性和简单性。
C
CyC2018 已提交
298

C
CyC2018 已提交
299
I/O 包和 NIO 已经很好地集成了,java.io.\* 已经以 NIO 为基础重新实现了,所以现在它可以利用 NIO 的一些特性。例如,java.io.\* 包中的一些类包含以块的形式读写数据的方法,这使得即使在面向流的系统中,处理速度也会更快。
C
CyC2018 已提交
300

C
CyC2018 已提交
301
## 通道与缓冲区
C
CyC2018 已提交
302

C
CyC2018 已提交
303
### 1. 通道
C
CyC2018 已提交
304

C
CyC2018 已提交
305
通道 Channel 是对原 I/O 包中的流的模拟,可以通过它读取和写入数据。
C
CyC2018 已提交
306

C
CyC2018 已提交
307
通道与流的不同之处在于,流只能在一个方向上移动(一个流必须是 InputStream 或者 OutputStream 的子类),而通道是双向的,可以用于读、写或者同时用于读写。
C
CyC2018 已提交
308 309 310

通道包括以下类型:

C
CyC2018 已提交
311 312 313 314
- FileChannel:从文件中读写数据;
- DatagramChannel:通过 UDP 读写网络中数据;
- SocketChannel:通过 TCP 读写网络中数据;
- ServerSocketChannel:可以监听新进来的 TCP 连接,对每一个新进来的连接都会创建一个 SocketChannel。
C
CyC2018 已提交
315

C
CyC2018 已提交
316
### 2. 缓冲区
C
CyC2018 已提交
317

C
CyC2018 已提交
318
发送给一个通道的所有数据都必须首先放到缓冲区中,同样地,从通道中读取的任何数据都要先读到缓冲区中。也就是说,不会直接对通道进行读写数据,而是要先经过缓冲区。
C
CyC2018 已提交
319 320 321 322 323

缓冲区实质上是一个数组,但它不仅仅是一个数组。缓冲区提供了对数据的结构化访问,而且还可以跟踪系统的读/写进程。

缓冲区包括以下类型:

C
CyC2018 已提交
324 325 326 327 328 329 330
- ByteBuffer
- CharBuffer
- ShortBuffer
- IntBuffer
- LongBuffer
- FloatBuffer
- DoubleBuffer
C
CyC2018 已提交
331

C
CyC2018 已提交
332
## 缓冲区状态变量
C
CyC2018 已提交
333

C
CyC2018 已提交
334 335 336
- capacity:最大容量;
- position:当前已经读写的字节数;
- limit:还可以读写的字节数。
C
CyC2018 已提交
337 338 339

状态变量的改变过程举例:

C
CyC2018 已提交
340
① 新建一个大小为 8 个字节的缓冲区,此时 position 为 0,而 limit = capacity = 8。capacity 变量不会改变,下面的讨论会忽略它。
C
CyC2018 已提交
341

C
CyC2018 已提交
342
<div align="center"> <img src="pics/1bea398f-17a7-4f67-a90b-9e2d243eaa9a.png"/> </div><br>
C
CyC2018 已提交
343

C
CyC2018 已提交
344
② 从输入通道中读取 5 个字节数据写入缓冲区中,此时 position 为 5,limit 保持不变。
C
CyC2018 已提交
345

C
CyC2018 已提交
346
<div align="center"> <img src="pics/80804f52-8815-4096-b506-48eef3eed5c6.png"/> </div><br>
C
CyC2018 已提交
347

C
CyC2018 已提交
348
③ 在将缓冲区的数据写到输出通道之前,需要先调用 flip() 方法,这个方法将 limit 设置为当前 position,并将 position 设置为 0。
C
CyC2018 已提交
349

C
CyC2018 已提交
350
<div align="center"> <img src="pics/952e06bd-5a65-4cab-82e4-dd1536462f38.png"/> </div><br>
C
CyC2018 已提交
351

C
CyC2018 已提交
352
④ 从缓冲区中取 4 个字节到输出缓冲中,此时 position 设为 4。
C
CyC2018 已提交
353

C
CyC2018 已提交
354
<div align="center"> <img src="pics/b5bdcbe2-b958-4aef-9151-6ad963cb28b4.png"/> </div><br>
C
CyC2018 已提交
355

C
CyC2018 已提交
356
⑤ 最后需要调用 clear() 方法来清空缓冲区,此时 position 和 limit 都被设置为最初位置。
C
CyC2018 已提交
357

C
CyC2018 已提交
358
<div align="center"> <img src="pics/67bf5487-c45d-49b6-b9c0-a058d8c68902.png"/> </div><br>
C
CyC2018 已提交
359

C
CyC2018 已提交
360
## 文件 NIO 实例
C
CyC2018 已提交
361

C
CyC2018 已提交
362
以下展示了使用 NIO 快速复制文件的实例:
C
CyC2018 已提交
363 364

```java
C
CyC2018 已提交
365
public static void fastCopy(String src, String dist) throws IOException {
C
CyC2018 已提交
366

C
CyC2018 已提交
367 368
    /* 获得源文件的输入字节流 */
    FileInputStream fin = new FileInputStream(src);
C
CyC2018 已提交
369

C
CyC2018 已提交
370 371
    /* 获取输入字节流的文件通道 */
    FileChannel fcin = fin.getChannel();
C
CyC2018 已提交
372

C
CyC2018 已提交
373 374
    /* 获取目标文件的输出字节流 */
    FileOutputStream fout = new FileOutputStream(dist);
C
CyC2018 已提交
375

C
CyC2018 已提交
376 377
    /* 获取输出字节流的文件通道 */
    FileChannel fcout = fout.getChannel();
C
CyC2018 已提交
378

C
CyC2018 已提交
379 380
    /* 为缓冲区分配 1024 个字节 */
    ByteBuffer buffer = ByteBuffer.allocateDirect(1024);
C
CyC2018 已提交
381

C
CyC2018 已提交
382
    while (true) {
C
CyC2018 已提交
383

C
CyC2018 已提交
384 385
        /* 从输入通道中读取数据到缓冲区中 */
        int r = fcin.read(buffer);
C
CyC2018 已提交
386

C
CyC2018 已提交
387 388 389 390
        /* read() 返回 -1 表示 EOF */
        if (r == -1) {
            break;
        }
C
CyC2018 已提交
391

C
CyC2018 已提交
392 393
        /* 切换读写 */
        buffer.flip();
C
CyC2018 已提交
394

C
CyC2018 已提交
395 396
        /* 把缓冲区的内容写入输出文件中 */
        fcout.write(buffer);
C
CyC2018 已提交
397

C
CyC2018 已提交
398 399 400
        /* 清空缓冲区 */
        buffer.clear();
    }
C
CyC2018 已提交
401 402 403
}
```

C
CyC2018 已提交
404
## 选择器
C
CyC2018 已提交
405

C
CyC2018 已提交
406
NIO 常常被叫做非阻塞 IO,主要是因为 NIO 在网络通信中的非阻塞特性被广泛使用。
C
CyC2018 已提交
407

C
CyC2018 已提交
408
NIO 实现了 IO 多路复用中的 Reactor 模型,一个线程 Thread 使用一个选择器 Selector 通过轮询的方式去监听多个通道 Channel 上的事件,从而让一个线程就可以处理多个事件。
C
CyC2018 已提交
409

C
CyC2018 已提交
410
通过配置监听的通道 Channel 为非阻塞,那么当 Channel 上的 IO 事件还未到达时,就不会进入阻塞状态一直等待,而是继续轮询其它 Channel,找到 IO 事件已经到达的 Channel 执行。
C
CyC2018 已提交
411

C
CyC2018 已提交
412
因为创建和切换线程的开销很大,因此使用一个线程来处理多个事件而不是一个线程处理一个事件,对于 IO 密集型的应用具有很好地性能。
C
CyC2018 已提交
413

C
CyC2018 已提交
414
应该注意的是,只有套接字 Channel 才能配置为非阻塞,而 FileChannel 不能,为 FileChannel 配置非阻塞也没有意义。
C
CyC2018 已提交
415

C
CyC2018 已提交
416
<div align="center"> <img src="pics/8fdbb8f5-2cf8-41dc-b5f1-99d98abb52d9.jpg"/> </div><br>
C
CyC2018 已提交
417

C
CyC2018 已提交
418
### 1. 创建选择器
C
CyC2018 已提交
419 420

```java
C
CyC2018 已提交
421
Selector selector = Selector.open();
C
CyC2018 已提交
422 423
```

C
CyC2018 已提交
424
### 2. 将通道注册到选择器上
C
CyC2018 已提交
425 426

```java
C
CyC2018 已提交
427
ServerSocketChannel ssChannel = ServerSocketChannel.open();
C
CyC2018 已提交
428
ssChannel.configureBlocking(false);
C
CyC2018 已提交
429
ssChannel.register(selector, SelectionKey.OP_ACCEPT);
C
CyC2018 已提交
430 431
```

C
CyC2018 已提交
432
通道必须配置为非阻塞模式,否则使用选择器就没有任何意义了,因为如果通道在某个事件上被阻塞,那么服务器就不能响应其它事件,必须等待这个事件处理完毕才能去处理其它事件,显然这和选择器的作用背道而驰。
C
CyC2018 已提交
433

C
CyC2018 已提交
434
在将通道注册到选择器上时,还需要指定要注册的具体事件,主要有以下几类:
C
CyC2018 已提交
435

C
CyC2018 已提交
436 437 438 439
- SelectionKey.OP_CONNECT
- SelectionKey.OP_ACCEPT
- SelectionKey.OP_READ
- SelectionKey.OP_WRITE
C
CyC2018 已提交
440

C
CyC2018 已提交
441
它们在 SelectionKey 的定义如下:
C
CyC2018 已提交
442

C
CyC2018 已提交
443
```java
C
CyC2018 已提交
444 445 446 447
public static final int OP_READ = 1 << 0;
public static final int OP_WRITE = 1 << 2;
public static final int OP_CONNECT = 1 << 3;
public static final int OP_ACCEPT = 1 << 4;
C
CyC2018 已提交
448 449
```

C
CyC2018 已提交
450
可以看出每个事件可以被当成一个位域,从而组成事件集整数。例如:
C
CyC2018 已提交
451 452

```java
C
CyC2018 已提交
453
int interestSet = SelectionKey.OP_READ | SelectionKey.OP_WRITE;
C
CyC2018 已提交
454 455
```

C
CyC2018 已提交
456
### 3. 监听事件
C
CyC2018 已提交
457 458

```java
C
CyC2018 已提交
459
int num = selector.select();
C
CyC2018 已提交
460 461
```

C
CyC2018 已提交
462
使用 select() 来监听到达的事件,它会一直阻塞直到有至少一个事件到达。
C
CyC2018 已提交
463

C
CyC2018 已提交
464
### 4. 获取到达的事件
C
CyC2018 已提交
465 466

```java
C
CyC2018 已提交
467 468 469 470 471 472 473 474 475 476
Set<SelectionKey> keys = selector.selectedKeys();
Iterator<SelectionKey> keyIterator = keys.iterator();
while (keyIterator.hasNext()) {
    SelectionKey key = keyIterator.next();
    if (key.isAcceptable()) {
        // ...
    } else if (key.isReadable()) {
        // ...
    }
    keyIterator.remove();
C
CyC2018 已提交
477 478 479
}
```

C
CyC2018 已提交
480
### 5. 事件循环
C
CyC2018 已提交
481

C
CyC2018 已提交
482
因为一次 select() 调用不能处理完所有的事件,并且服务器端有可能需要一直监听事件,因此服务器端处理事件的代码一般会放在一个死循环内。
C
CyC2018 已提交
483 484

```java
C
CyC2018 已提交
485 486 487 488 489 490 491 492 493 494 495 496 497
while (true) {
    int num = selector.select();
    Set<SelectionKey> keys = selector.selectedKeys();
    Iterator<SelectionKey> keyIterator = keys.iterator();
    while (keyIterator.hasNext()) {
        SelectionKey key = keyIterator.next();
        if (key.isAcceptable()) {
            // ...
        } else if (key.isReadable()) {
            // ...
        }
        keyIterator.remove();
    }
C
CyC2018 已提交
498
}
C
CyC2018 已提交
499 500
```

C
CyC2018 已提交
501
## 套接字 NIO 实例
C
CyC2018 已提交
502 503

```java
C
CyC2018 已提交
504
public class NIOServer {
C
CyC2018 已提交
505

C
CyC2018 已提交
506
    public static void main(String[] args) throws IOException {
C
CyC2018 已提交
507

C
CyC2018 已提交
508
        Selector selector = Selector.open();
C
CyC2018 已提交
509

C
CyC2018 已提交
510 511 512
        ServerSocketChannel ssChannel = ServerSocketChannel.open();
        ssChannel.configureBlocking(false);
        ssChannel.register(selector, SelectionKey.OP_ACCEPT);
C
CyC2018 已提交
513

C
CyC2018 已提交
514 515 516
        ServerSocket serverSocket = ssChannel.socket();
        InetSocketAddress address = new InetSocketAddress("127.0.0.1", 8888);
        serverSocket.bind(address);
C
CyC2018 已提交
517

C
CyC2018 已提交
518
        while (true) {
C
CyC2018 已提交
519

C
CyC2018 已提交
520 521 522
            selector.select();
            Set<SelectionKey> keys = selector.selectedKeys();
            Iterator<SelectionKey> keyIterator = keys.iterator();
C
CyC2018 已提交
523

C
CyC2018 已提交
524
            while (keyIterator.hasNext()) {
C
CyC2018 已提交
525

C
CyC2018 已提交
526
                SelectionKey key = keyIterator.next();
C
CyC2018 已提交
527

C
CyC2018 已提交
528
                if (key.isAcceptable()) {
C
CyC2018 已提交
529

C
CyC2018 已提交
530
                    ServerSocketChannel ssChannel1 = (ServerSocketChannel) key.channel();
C
CyC2018 已提交
531

C
CyC2018 已提交
532 533 534
                    // 服务器会为每个新连接创建一个 SocketChannel
                    SocketChannel sChannel = ssChannel1.accept();
                    sChannel.configureBlocking(false);
C
CyC2018 已提交
535

C
CyC2018 已提交
536 537
                    // 这个新连接主要用于从客户端读取数据
                    sChannel.register(selector, SelectionKey.OP_READ);
C
CyC2018 已提交
538

C
CyC2018 已提交
539
                } else if (key.isReadable()) {
C
CyC2018 已提交
540

C
CyC2018 已提交
541 542 543 544
                    SocketChannel sChannel = (SocketChannel) key.channel();
                    System.out.println(readDataFromSocketChannel(sChannel));
                    sChannel.close();
                }
C
CyC2018 已提交
545

C
CyC2018 已提交
546 547 548 549
                keyIterator.remove();
            }
        }
    }
C
CyC2018 已提交
550

C
CyC2018 已提交
551
    private static String readDataFromSocketChannel(SocketChannel sChannel) throws IOException {
C
CyC2018 已提交
552

C
CyC2018 已提交
553 554
        ByteBuffer buffer = ByteBuffer.allocate(1024);
        StringBuilder data = new StringBuilder();
C
CyC2018 已提交
555

C
CyC2018 已提交
556
        while (true) {
C
CyC2018 已提交
557

C
CyC2018 已提交
558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573
            buffer.clear();
            int n = sChannel.read(buffer);
            if (n == -1) {
                break;
            }
            buffer.flip();
            int limit = buffer.limit();
            char[] dst = new char[limit];
            for (int i = 0; i < limit; i++) {
                dst[i] = (char) buffer.get(i);
            }
            data.append(dst);
            buffer.clear();
        }
        return data.toString();
    }
C
CyC2018 已提交
574
}
C
CyC2018 已提交
575 576 577
```

```java
C
CyC2018 已提交
578 579 580 581 582 583 584 585 586
public class NIOClient {

    public static void main(String[] args) throws IOException {
        Socket socket = new Socket("127.0.0.1", 8888);
        OutputStream out = socket.getOutputStream();
        String s = "hello world";
        out.write(s.getBytes());
        out.close();
    }
C
CyC2018 已提交
587 588 589
}
```

C
CyC2018 已提交
590
## 内存映射文件
C
CyC2018 已提交
591

C
CyC2018 已提交
592
内存映射文件 I/O 是一种读和写文件数据的方法,它可以比常规的基于流或者基于通道的 I/O 快得多。
C
CyC2018 已提交
593

C
CyC2018 已提交
594
向内存映射文件写入可能是危险的,只是改变数组的单个元素这样的简单操作,就可能会直接修改磁盘上的文件。修改数据与将数据保存到磁盘是没有分开的。
C
CyC2018 已提交
595

C
CyC2018 已提交
596
下面代码行将文件的前 1024 个字节映射到内存中,map() 方法返回一个 MappedByteBuffer,它是 ByteBuffer 的子类。因此,可以像使用其他任何 ByteBuffer 一样使用新映射的缓冲区,操作系统会在需要时负责执行映射。
C
CyC2018 已提交
597 598

```java
C
CyC2018 已提交
599
MappedByteBuffer mbb = fc.map(FileChannel.MapMode.READ_WRITE, 0, 1024);
C
CyC2018 已提交
600 601
```

C
CyC2018 已提交
602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620
## 对比

NIO 与普通 I/O 的区别主要有以下两点:

- NIO 是非阻塞的;
- NIO 面向块,I/O 面向流。

# 八、参考资料

- Eckel B, 埃克尔, 昊鹏, 等. Java 编程思想 [M]. 机械工业出版社, 2002.
- [IBM: NIO 入门](https://www.ibm.com/developerworks/cn/education/java/j-nio/j-nio.html)
- [Java NIO Tutorial](http://tutorials.jenkov.com/java-nio/index.html)
- [Java NIO 浅析](https://tech.meituan.com/nio.html)
- [IBM: 深入分析 Java I/O 的工作机制](https://www.ibm.com/developerworks/cn/java/j-lo-javaio/index.html)
- [IBM: 深入分析 Java 中的中文编码问题](https://www.ibm.com/developerworks/cn/java/j-lo-chinesecoding/index.html)
- [IBM: Java 序列化的高级认识](https://www.ibm.com/developerworks/cn/java/j-lo-serial/index.html)
- [NIO 与传统 IO 的区别](http://blog.csdn.net/shimiso/article/details/24990499)
- [Decorator Design Pattern](http://stg-tud.github.io/sedc/Lecture/ws13-14/5.3-Decorator.html#mode=document)
- [Socket Multicast](http://labojava.blogspot.com/2012/12/socket-multicast.html)
C
CyC2018 已提交
621 622 623 624




C
CyC2018 已提交
625
</br><div align="center"> <img src="https://cyc-1256109796.cos.ap-guangzhou.myqcloud.com/%E5%85%AC%E4%BC%97%E5%8F%B7%20%E6%B5%B7%E6%8A%A5.png" width="500px"> </div></br>