[关闭]
@lemonguge 2015-07-03T08:23:53.000000Z 字数 6630 阅读 452

Java nio(四)

NIO


管道Pipe

在之前介绍了关于Socket流的通道和文件通道,这里将介绍nio的管道Pipe。查看API可以发现该类只有有三个方法,不过应该注意到,它具有两个嵌套类:Pipe.SinkChannelPipe.SourceChannel

简单的说,数据会被写到sink 通道,从source 通道读取,以下是Pipe原理的图示:

管道的原理图

现在来明确使用管道的基本思路,首先我们需要创建管道,通过Pipe的静态open()方法打开管道。

向管道写入数据,需要访问Pipe.SinkChannel,通过管道的sink()获取可写入管道。我们可以通过该管道的write()方法将缓冲区的数据写入。

从管道读取数据,需要访问Pipe.SourceChannel,通过管道的source()获取可读取管道。可以通过该管道的read()方法将数据读取到缓冲区。当该通道已到达流的末尾,则返回-1。

以下是一个完整的管道示例:

  1. import java.io.IOException;
  2. import java.nio.ByteBuffer;
  3. import java.nio.channels.Pipe;
  4. public class PipeDemo {
  5. public static void main(String[] args) throws IOException {
  6. int length = 1024;
  7. Pipe pipe = Pipe.open();
  8. Pipe.SinkChannel sink = pipe.sink();
  9. Pipe.SourceChannel source = pipe.source();
  10. ByteBuffer buf = ByteBuffer.wrap("Hello World!".getBytes());
  11. while (buf.hasRemaining())
  12. sink.write(buf);
  13. // 如果不关闭可写入管道,可读取管道的read方法会一直阻塞,不会返回-1
  14. sink.close();
  15. buf = ByteBuffer.allocate(length);
  16. byte[] dst = new byte[length];
  17. int size = 0;
  18. while ((size = source.read(buf)) != -1) {
  19. buf.flip();
  20. buf.get(dst, 0, size);
  21. System.out.println(new String(dst, 0, size));
  22. buf.clear();
  23. }
  24. source.close();
  25. }
  26. } /* Output:
  27. Hello World!
  28. *///:~

文件锁FileLock

文件锁就像是同步时的对象锁,它们是劝告式(advisory)的锁。劝告式的锁又称询问式的锁,这种类型的锁将拥有互斥的效果。锁的效果与操作系统有关,锁还有另一种效果,强制性(mandatory),当某个进程获得锁后,操作系统将保证其他进程无法操作共享资源。

对于文件锁FileLock的两种不同的锁定方式,可以文件锁分为两种:

通过查看关于FileChannel文件通道的API,发现有四种获取锁的方式:

lock()方法为阻塞式方法、tryLock()方法是非阻塞的方法,如果想通过tryLock()获取锁,可以放在do..while..进行循环判断。当拥有锁后,可以调用FileLock对象的release()释放锁,尝试获得锁的其他任何线程都有机会获得它。

以下是一个关于文件锁的示例:

  1. import java.io.BufferedReader;
  2. import java.io.File;
  3. import java.io.FileInputStream;
  4. import java.io.IOException;
  5. import java.io.InputStreamReader;
  6. import java.io.RandomAccessFile;
  7. import java.nio.channels.FileChannel;
  8. import java.nio.channels.FileLock;
  9. class LockFile implements Runnable {
  10. private FileLock lock;
  11. private RandomAccessFile raf;
  12. public LockFile(FileLock lock, RandomAccessFile raf) {
  13. this.lock = lock;
  14. this.raf = raf;
  15. }
  16. @Override
  17. public void run() {
  18. try {
  19. // 判断锁的类型,true为共享锁
  20. System.out.println(lock.isShared() ? "共享锁" : "独占锁");
  21. Thread.sleep(3000);
  22. System.out.println("释放锁");
  23. // 释放锁
  24. lock.release();
  25. raf.close();
  26. } catch (IOException e) {
  27. e.printStackTrace();
  28. } catch (InterruptedException e) {
  29. e.printStackTrace();
  30. }
  31. }
  32. }
  33. // 普通的文件读取
  34. class FileRead implements Runnable {
  35. private File file;
  36. public FileRead(File file) {
  37. this.file = file;
  38. }
  39. @Override
  40. public void run() {
  41. try {
  42. System.out.println("读取文件");
  43. FileInputStream fis = new FileInputStream(file);
  44. BufferedReader br = new BufferedReader(new InputStreamReader(fis));
  45. String line = null;
  46. while ((line = br.readLine()) != null)
  47. System.out.println(line);
  48. br.close();
  49. } catch (IOException e) {
  50. e.printStackTrace();
  51. }
  52. }
  53. }
  54. public class FileLockDemo {
  55. public static void main(String[] args) throws InterruptedException, IOException {
  56. // 在当前项目下有file.txt文件,文件有两行内容
  57. // 分别为“abc123”和“你好”
  58. File file = new File("file.txt");
  59. RandomAccessFile raf = new RandomAccessFile(file, "rw");
  60. // 查看传入共享锁和独占锁的输出结果
  61. new Thread(new LockFile(getLock(raf, false), raf)).start();
  62. // 当传入独占锁时,设置睡眠时间为1000和4000的输出结果
  63. Thread.sleep(1000);
  64. new Thread(new FileRead(file)).start();
  65. }
  66. // 设置isShared为true将获得共享锁、false获得独占锁
  67. private static FileLock getLock(RandomAccessFile raf, boolean isShared) throws IOException {
  68. FileChannel fileChnl = raf.getChannel();
  69. if (isShared)
  70. return fileChnl.lock(0, fileChnl.size(), true); // 同样锁定整个文件
  71. else
  72. // 获取独占锁,必须为可写入通道
  73. return fileChnl.lock();
  74. }
  75. } /* Output:
  76. 独占锁
  77. 读取文件
  78. java.io.IOException: 另一个程序已锁定文件的一部分,进程无法访问。
  79. at java.io.FileInputStream.readBytes(Native Method)
  80. at java.io.FileInputStream.read(FileInputStream.java:272)
  81. at sun.nio.cs.StreamDecoder.readBytes(StreamDecoder.java:283)
  82. at sun.nio.cs.StreamDecoder.implRead(StreamDecoder.java:325)
  83. at sun.nio.cs.StreamDecoder.read(StreamDecoder.java:177)
  84. at java.io.InputStreamReader.read(InputStreamReader.java:184)
  85. at java.io.BufferedReader.fill(BufferedReader.java:154)
  86. at java.io.BufferedReader.readLine(BufferedReader.java:317)
  87. at java.io.BufferedReader.readLine(BufferedReader.java:382)
  88. at nio.FileRead.run(FileLockDemo.java:50)
  89. at java.lang.Thread.run(Thread.java:744)
  90. 释放锁
  91. *///:~

当传入共享锁,睡眠时间为1000ms时,会显示以下输出结果:

  1. 共享锁
  2. 读取文件
  3. abc123
  4. 你好
  5. 释放锁

当传入独占锁,睡眠时间为4000ms时,则会显示以下输出结果:

  1. 独占锁
  2. 释放锁
  3. 读取文件
  4. abc123
  5. 你好

如果获取一个独占锁,那么其他人就不能获得同一个文件或者文件的一部分上的锁。如果获得一个共享锁,那么其他人可以获得同一个文件或者文件一部分上的共享锁,但是不能获得独占锁。

以下是一个线程获得了共享锁后,另一个线程想获得独占锁的示例:

  1. import java.io.File;
  2. import java.io.FileInputStream;
  3. import java.io.IOException;
  4. import java.io.RandomAccessFile;
  5. import java.nio.channels.FileLock;
  6. class ShareLock implements Runnable {
  7. private File file;
  8. public ShareLock(File file) {
  9. this.file = file;
  10. }
  11. @Override
  12. public void run() {
  13. FileInputStream fis;
  14. try {
  15. fis = new FileInputStream(file);
  16. System.out.println("获取共享锁");
  17. FileLock lock = fis.getChannel().lock(0, fis.available(), true);
  18. Thread.sleep(3000);
  19. lock.release();
  20. System.out.println("释放共享锁");
  21. } catch (IOException e) {
  22. e.printStackTrace();
  23. } catch (InterruptedException e) {
  24. e.printStackTrace();
  25. }
  26. }
  27. }
  28. public class SharedLockDemo {
  29. public static void main(String[] args) throws IOException, InterruptedException {
  30. File file = new File("file.txt");
  31. new Thread(new ShareLock(file)).start();
  32. RandomAccessFile raf = new RandomAccessFile(file, "rw");
  33. // 保证会先获得共享锁
  34. Thread.sleep(1000);
  35. System.out.println("获取独占锁");
  36. // 获得独占锁
  37. raf.getChannel().lock();
  38. // 关闭文件通道,也会使FileLock的生命周期结束
  39. raf.close();
  40. }
  41. } /* Output:
  42. 获取共享锁
  43. 获取独占锁
  44. Exception in thread "main" java.nio.channels.OverlappingFileLockException
  45. at sun.nio.ch.SharedFileLockTable.checkList(FileLockTable.java:255)
  46. at sun.nio.ch.SharedFileLockTable.add(FileLockTable.java:152)
  47. at sun.nio.ch.FileChannelImpl.lock(FileChannelImpl.java:1011)
  48. at java.nio.channels.FileChannel.lock(FileChannel.java:1052)
  49. at nio.SharedLockDemo.main(SharedLockDemo.java:44)
  50. 释放共享锁
  51. *///:~

文件锁定并不总是出于保护数据的目的,可能临时锁定一个文件以保证特定的写操作成为原子的,而不会有其他程序的干扰。


字符集Charset

在我的Java I/O系列的第三篇文章中转换流部分,曾经提到nio中的字符集Charset类,当时通过该类获得系统的默认编码。接下来我要讲述的类都在java.nio.charset包中,查看API可以发现是这么描述字符集的:

16位的Unicode代码单元序列(字符序列)和字节序列之间的指定映射关系。

Java语言被定义为基于Unicode,即使你把类名写成中文也同样能运行。

  1. public class 你好 {
  2. public static void main(String[] args) {
  3. System.out.println("你好");
  4. }
  5. } /* Output:
  6. 你好
  7. *///:~

解码与编码

在Java I/O中提到过解码与编码,转换流InputStreamReaderOutputStreamWriter就分别对应着解码和编码。不过在Java的一切都是对象的思想下,编码和解码也变成了对象,我们把编码称为CharsetEncoder编码器,解码称为CharsetDecoder解码器。简单的来说,解码用于读取,编码用于写入

我们可以通过字符集Charset类的静态forName()方法获取指定字符集名称的字符集。

通过此字符集,可以通过newDecoder()创建一个解码器,通过newEncoder()方法创建一个编码器。

可以通过CharsetDecoder解码器的decode()方法将字节缓冲区解码到新分配的字符缓冲区,然后我们可以处理字符了。

如果想要写好数据,必须使用CharsetEncoder编码器的encode()方法将字符缓冲区编码到新分配的字节缓冲区。

解码器和编码器一般可以和标准I/O流结合使用,即在构造转换流时当作参数传入。


后记:nio库有大量的特性,它为我们最熟悉的标准I/O过程也带来了新的活力,通道和缓冲区可以做的事情几乎都可以用原来的面向流的类来完成。但是通道和缓冲区可以以接近系统所允许的最大速度完成这些相同的旧操作。知识点挺多的,不过终于写完啦,写于2015年7月3日。

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注