【Java】NIO和BIO有什么区别?答案:天壤之别

第一时间阅读最新文章

【Java】NIO和BIO有什么区别?答案:天壤之别
【Java】NIO不简单呐,Channel、Buffer、Selector

一、什么是NIO

1.概念

NIO是java1.4中引入的,被称为new I/O,也有说是non-blocking I/O,NIO被成为同步非阻塞的IO。

2.跟BIO流的区别

  1. BIO是面向流的,NIO是面向块(缓冲区)的
  2. BIO的流都是同步阻塞的,而NIO同步非阻塞的
  3. NIO会等待数据全部传输过来再让线程处理,BIO是直接让线程等待。
  4. NIO有选择器,而BIO没有。

二、如何使用

这里以文件复制为例

1.代码

public class test {
    public static void main(String[] args){
        try{
            File inFile=new File("C:\\Users\\Administrator\\Desktop\\study.PNG");
            File outFile=new File("C:\\Users\\Administrator\\Desktop\\study1.PNG");
            FileInputStream fileInputStream=new FileInputStream(inFile);
            FileOutputStream fileOutputStream=new FileOutputStream(outFile);
            /**
             * RandomAccessFile accessFile=new RandomAccessFile(inFile,"wr");
             *  FileChannel inFileChannel=accessFile.getChannel();
             *  和下面两行代码是一样的,都是可以拿到FileChannel
             */
            FileChannel inFileChannel=fileInputStream.getChannel();
            FileChannel outFileChannel=fileOutputStream.getChannel();

            ByteBuffer buffer=ByteBuffer.allocate(1024*1024);

            while (inFileChannel.read(buffer)!=-1){
                buffer.flip();
                outFileChannel.write(buffer);
                buffer.clear();
            }
            inFileChannel.close();
            outFileChannel.close();
            fileInputStream.close();
            fileOutputStream.close();
        }catch (Exception e){}

    }
}

我的桌面上的确多了一张一模一样的图片

2.解释

使用NIO的话,需要注意几个步骤:

  1. 打开流
  2. 获取通道
  3. 创建Buffer
  4. 切换到读模式 buffer.flip()
  5. 切换到写模式 buffer.clear();
    其实这里也看不出来它是怎么使用缓冲区的,上面这段代码中的while循环的作用和下面的代码是一样的
 while ((i=fileInputStream.read())!=-1){
                fileOutputStream.write(i);
          }

那我们来检验一下它们的性能吧

3.BIO和NIO的性能区别

代码

我整理了一下代码,把复制文件的功能都整合在方法里了,只需要传入两个文件对象的就可以了

 private static void NIOTest(File inFile,File outFile) {
        long startTime = System.currentTimeMillis();
        try{
            //创建文件流
            FileInputStream fileInputStream=new FileInputStream(inFile);
            FileOutputStream fileOutputStream=new FileOutputStream(outFile);
            //获取通道
            FileChannel inFileChannel=fileInputStream.getChannel();
            FileChannel outFileChannel=fileOutputStream.getChannel();
            //开辟缓冲区,设置缓冲区大小
            ByteBuffer buffer=ByteBuffer.allocate((int)inFile.length());
            //读取
            while (inFileChannel.read(buffer)!=-1){
            //写入
                buffer.flip();
                outFileChannel.write(buffer);
                buffer.clear();
            }
            //关闭通道和流
            inFileChannel.close();
            outFileChannel.close();
            fileInputStream.close();
            fileOutputStream.close();
        }catch (Exception e){}
        System.out.println("NIO: "+(System.currentTimeMillis()-startTime)+"ms");
    }
   private static void IOTest(File inFile,File outFile) {
        long startTime = System.currentTimeMillis();
        try{
            FileInputStream fileInputStream=new FileInputStream(inFile);
            FileOutputStream fileOutputStream=new FileOutputStream(outFile);
            int i=0;
            byte[] bytes=new byte[(int)inFile.length()];
            while (fileInputStream.read(bytes)!=-1){
                fileOutputStream.write(bytes);
            }
            fileInputStream.close();
            fileOutputStream.close();
        }catch (Exception e){}
        System.out.println("IO: "+(System.currentTimeMillis()-startTime)+"ms");
    }

结果

IO: 2ms
NIO: 5ms

这个文件才一百k,换个大的文件试一下,这里我用了一个六十多M的压缩包

IO: 712ms
NIO: 737ms

结论

根据我的多次实验,发现,在开辟的缓存区大小一样的情况下,NIO并不比IO快
但是,当在处理比较大的文件时,缓存区的大小设置为‘文件大小/8~64’时,NIO比IO快,当然这个范围也不是准确的,有兴趣的可以一个一个去测试,看看那个比较快
而在处理比较小的文件时,无论缓存区的大小设置为多少,NIO都比IO慢
我是根据测试结果来做的的总结,可能文件再大一点就会不一样了,如果有大佬做了其他测试,得出了新结论的话,可以告诉我,我把结论改一下。

——————————————————

后续的补充:
在文件传输方面体现不出NIO的优势,在本地传输文件的话,不存在多个请求,所以这里的BIO和NIO差别不会太大。只有在同时有多个请求过来的情况下,才会体现出NIO的优势,所以NIO一般都是在网络编程中提到的,本地的磁盘IO使用NIO效果甚微,后续有机会的话,我会出一期网络编程的NIO,彻底把Java的NIO讲清楚。

4.selector的使用

只有使用套接字(ServerSocketChannel/SocketChannel)才能真正发挥NIO的非阻塞,但是,我对套接字这些不是很懂,就只好贴个别人写好的客户机和服务器的网址了
NIO的Selector介绍和例子代码
就不献丑了

步骤

在这里插入图片描述

画了个图来表示,这是关于selector的配置流程,在循环中根据不同key值所进行的操作,跟上面文件复制的例子差不多了,只不过这里的Channel是通过 key.channel()获得

三、总结

使用NIO比使用BIO繁琐,除开Selector,NIO与BIO之间不同的就是缓冲区有所不同BIO的缓冲区就是一个单纯的数组,而NIO的缓冲区则有很多定位的属性,操作起来也比BIO的数组麻烦
而Selector则让NIO实现了多路复用的功能,通过SelectorKey的迭代器就可以实现对多个Channel的不断轮询,从而实现多路复用

最后,我也有一个问题,一直没有搞懂,就是NIO的非阻塞到底体现在哪里?
是它底层的实现和IO流的不一样?还是因为有了Selector?

——————————————————————

回答:其实这里涉及到了NIO和BIO的底层是怎么运行的,我当时没有搞明白,就留下了这个疑惑,不过现在我就有了答案。拿网络IO来举例,BIO的话,每次网络请求过来之后,服务器都是会为这个请求创建一个线程,这个线程会一直等待这个请求后续的数据,等处理完成后才会销毁这个线程;而NIO,当每次网络请求过来时,服务器不会马上创建一个线程去处理这个请求,而是会交给一个Selector线程,只有这个请求后续的数据全部传输过来后,Selector才会去通知其他其他线程或者创建一个线程来处理这个请求。

——————————————————————————————

你知道的越多,不知道的就越多。

如果本文章内容有问题,请直接评论或者私信我。如果觉得我写得还不错的话,点个赞也是对我的支持哦

未经允许,不得转载!

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

推荐阅读更多精彩内容