Java-nio

nio原理学习 nio简介

nio 是New IO 的简称,在jdk1.4 里提供的新api 。Sun 官方标榜的特性如下: 为所有的原始类型提供(Buffer)缓存支持。字符集编码解码解决方案。 Channel :一个新的原始I/O 抽象。 支持锁和内存映射文件的文件访问接口。 提供多路(non-bloking) 非阻塞式的高伸缩性网络I/O 。
传统的I/O
  • 使用传统的I/O程序读取文件内容, 并写入到另一个文件(或Socket), 如下程序:
    • File.read(fileDesc, buf, len);
    • Socket.send(socket, buf, len);
  • 会有较大的性能开销, 主要表现在一下两方面:
    1. 上下文切换(context switch), 此处有4次用户态和内核态的切换
    2. Buffer内存开销, 一个是应用程序buffer, 另一个是系统读取buffer以及socket buffer其运行示意图如下
      ?
Java-nio
文章图片
mark
  1. 先将文件内容从磁盘中拷贝到操作系统buffer
  1. 再从操作系统buffer拷贝到程序应用buffer
  1. 从程序buffer拷贝到socket buffer
  2. 从socket buffer拷贝到协议引擎.
NIO
  • NIO技术省去了将操作系统的read buffer拷贝到程序的buffer, 以及从程序buffer拷贝到socket buffer的步骤, 直接将 read buffer 拷贝到 socket buffer. java 的 FileChannel.transferTo() 方法就是这样的实现, 这个实现是依赖于操作系统底层的sendFile()实现的.
  • publicvoid transferTo(long position, long count, WritableByteChannel target);
    他的底层调用的是系统调用sendFile()方法
    sendfile(int out_fd, int in_fd, off_t *offset, size_t count);
    如下图
    Java-nio
    文章图片
    mark
传统socket和socket nio代码
  • 传统socket
    • server
    • public static void main(String args[]) throws Exception { // 监听端口 ServerSocket server_socket = new ServerSocket(2000); System.out.println("等待,端口为:" + server_socket.getLocalPort()); while (true) { // 阻塞接受消息 Socket socket = server_socket.accept(); // 打印链接信息 System.out.println("新连接: " + socket.getInetAddress() + ":" + socket.getPort()); // 从socket中获取流 DataInputStream input = new DataInputStream(socket.getInputStream()); // 接收数据 byte[] byteArray = new byte[4096]; while (true) { int nread = input.read(byteArray, 0, 4096); System.out.println(new String(byteArray, "UTF-8")); if (-1 == nread) { break; }} socket.close(); System.out.println("Connection closed by client"); } }

    • client
    • public static void main(String[] args) throws Exception { long start = System.currentTimeMillis(); // 创建socket链接 Socket socket = new Socket("localhost", 2000); System.out.println("Connected with server " + socket.getInetAddress() + ":" + socket.getPort()); // 读取文件 FileInputStream inputStream = new FileInputStream("C:/sss.txt"); // 输出文件 DataOutputStream output = new DataOutputStream(socket.getOutputStream()); // 缓冲区4096K byte[] b = new byte[4096]; // 传输长度 long read = 0, total = 0; // 读取文件,写到socketio中 while ((read = inputStream.read(b)) >= 0) { total = total + read; output.write(b); } // 关闭 output.close(); socket.close(); inputStream.close(); // 打印时间 System.out.println("bytes send--" + total + " and totaltime--" + (System.currentTimeMillis() - start)); }

  • socket nio 代码
    • server
      • public static void main(String[] args) throws IOException { // 创建socket channel ServerSocketChannel serverSocketChannel = ServerSocketChannel.open(); ServerSocket ss = serverSocketChannel.socket(); ss.setReuseAddress(true); // 地址重用 ss.bind(new InetSocketAddress("localhost", 9026)); // 绑定地址 System.out.println("监听端口 : " + new InetSocketAddress("localhost", 9026).toString()); // 分配一个新的字节缓冲区 ByteBuffer dst = ByteBuffer.allocate(4096); // 读取数据 while (true) { SocketChannel channle = serverSocketChannel.accept(); // 接收数据 System.out.println("Accepted : " + channle); channle.configureBlocking(true); // 设置阻塞,接不到就停 int nread = 0; while (nread != -1) { try { nread = channle.read(dst); // 往缓冲区里读 byte[] array = dst.array(); //将数据转换为array //打印 String string = new String(array, 0, dst.position()); System.out.print(string); dst.clear(); } catch (IOException e) { e.printStackTrace(); nread = -1; } } } }

    • client
      • public static void main(String[] args) throws IOException { long start = System.currentTimeMillis(); // 打开socket的nio管道 SocketChannel sc = SocketChannel.open(); sc.connect(new InetSocketAddress("localhost", 9026)); // 绑定相应的ip和端口 sc.configureBlocking(true); // 设置阻塞 // 将文件放到channel中 FileChannel fc = new FileInputStream("C:/sss.txt").getChannel(); // 打开文件管道 //做好标记量 long size = fc.size(); int pos = 0; int offset = 4096; long curnset = 0; long counts = 0; //循环写 while (pos

        【Java-nio】?

    推荐阅读