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其运行示意图如下

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);

    如下图

    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<size) {
                  curnset = fc.transferTo(pos, 4096, sc);// 把文件直接读取到socket chanel中,返回文件大小
                  pos+=offset;
                  counts+=curnset;
              }
              //关闭
              fc.close();
              sc.close();
              //打印传输字节数
              System.out.println(counts);
              // 打印时间
              System.out.println("bytes send--" + counts + " and totaltime--"
                              + (System.currentTimeMillis() - start));
          }
        

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

推荐阅读更多精彩内容