• UID19083
  • 登录2017-12-07
  • 粉丝0
  • 关注0
  • 发帖11
  • 主页
  • 金币52枚
qq_rFRhEoj14991 发布于2017-07-14 11:33
0/196

【转】全栈知识分享,NodeJS异步I/O解析

楼层直达
在现在的项目开发中,任何一个大型项目绝对不是简简单单的采用一个种语言和一种框架,因为每种语言和框架各有优势,与其死守一个,不与取各家之所长,依次得到一个高性能、搞扩展的产品。
对于一个.NET开发者,尤其是主要从事Web开发的.NET程序员,个人觉得有必要学习一门性能优越的Web平台开发语言。一个开发者不能简简单单的只学习一门语言,思维应该开阔,从各个方面去看待同样的一个问题,这样或许会得到另一番效果和见解,个人认为应该学习一下其他的语言,这样有利于我们对比语言的优势和缺点,例如java、nodejs、python等等。
对于Nodejs这个JavaScript平台,个人觉得.NET程序员有必要学习一下,因为学习NodeJS有助于我们构建一个高性能的Web平台。
NodeJS具有事件驱动、非阻塞I/O等特点,可以很好的处理I/O操作。Node面向网络且擅长并行I/O,能够有效地组织起更多的硬件资源。

这篇博客就来简单的介绍一下NodeJS的异步I/O特点。关于node.js的入门了解,更多讨论群550392000~

一.NodeJS概述


要学习一个语言或者平台,我们首先应该知道其定义,依据定义来扩展我们的学习思路。Node的定义:”一个大奖在Chrome JavaScript运行时上的平台,用于构建高速、可伸缩的网络程序。NodeJS作为一个异步事件驱动的JavaScript运行时,旨在构建可扩展的网络应用程序。“有关nodejs的背景介绍和安装方法,这里就没有必要介绍了,因为对于nodejs的安装是比较简单,所以在这里赘述就有些显得浪费时间。
学习完Node的定义特点,可能很多人都会好奇这个平台的适用场景是什么,以便在实际的项目开发中应用,不然学习这个就没有意义。主要的应用场景:前后端编程语言环境统一;高性能I/O用于实时应用;并行I/O使得使用者可以更高效地利用分布式环境;并行I/O有效利用稳定接口提升Web渲染能力;云平台的支持;游戏开发(这可能是很多开发者在意的,毕竟现在的游戏开发火热程度已经到了无以附加的地步);工具类应用,与较多的工具方法,使得开发效率大大的提升。
NodeJS异步I/O模型的基本要素:事件循环、观察者、请求对象、I/O线程池这四个共同构成。接下来我们具体了解一下这些知识。


二.NodeJS异步I/O解析
对于Nginx服务器,很多人都是比较的熟悉,Nginx采用纯C编写而成,用于做Web服务器,在反向代理和负载均衡等服务方面有很好的优势。Node与Nginx服务器有着相似的地方,都是采用事件驱动。
浏览器中JavaScript在单线程上执行,而且还与UI渲染共用一个线程,JavaScript在执行的时候UI渲染和响应应是出于停滞状态。(如果脚本执行的时间超过100毫秒,用户就会感到页面卡顿)。遇到这些情况,我们就会想到异步的方式消除这些等待的问题,对于异步和同步的概念就不做介绍了。
接下来我们具体的来了解一下NodeJS的事件驱动和非阻塞I/O这些特点,了解这些对于我们更好的学习NodeJS开发和构建高性能的Web平台有更加深远的意义。
1、I/O操作概述
I/O操作对于任何一个开发者来说都不会陌生,现在我们就简单的谈一下NodeJS的I.O操作。I/O操作分为:单线程串行依次执行;多线程并行执行。
这两种方式各有优势和缺点,多线程的代价在于创建线程和执行期线程上下文切换的开销较大,并且多线程面临锁、状态同步的问题。
单线程安装顺序执行,在执行中任何一个稍慢都会导致后续执行代码阻塞。对于任务的串行执行(概念上类似于同步执行)和任务的并行执行的描述有如下图:
在NodeJS中利用单线程,远离死锁、状态同步问题,利用异步I/O,让单线程远离阻塞,以便更好的使用CPU。异步I/O是期望I/O的调用不再阻塞后续运算,将原有等待I/O完成这段时间分配给其他需要的业务去执行。
很多时候一些开发者对异步/同步和阻塞/非阻塞的概念有些分不清,这两者没有什么关联。阻塞I/O是调用之后一定要等到系统内核层面完成所有操作后,调用才结束。非阻塞I/O是在调用后立即返回。


2.NodeJS异步I/O解析
事件循环:在进程启动时,Node会创建一个类似于while(true)的循环,每执行一次循环体的过程称为Tick,每个Tick的过程就是查看是否有时间待处理。
观察者:每个时间循环中有一个或多个观察者,判断是否有事件要处理的过程就是向这些观察者询问是否又要处理的事件。
请求对象:从JavaScript发起调用到内核执行完I/O操作的过渡过程中,存在一种中间产物,就是请求对象。
I/O线程池:组装好请求、送入I/O线程池等待执行,完成第一步I/O操作,进入第二部分回调通知。(在Windows中,线程池中的I/O操作调用完毕之后,会将获取的结果存在req->result属性上,然后调用PostQueuedCompletionStatus()通知IOCP,告知当前对象操作已经完成。)
异步I/O有如下图:

图片:640.jpg




三.NodeJS异步编程实例
前面介绍了异步I/O的相关概念,这里提供一个异步I/O操作的实例:
var config = require('./config.json');
var fs = require("fs");
var http = require('http');
var url_module = require("url");


http.createServer(function (request, response) {
   var key = url_module.parse(request.url).query.replace('key=', '');
   switch (request.method) {
       case 'GET':  // Asynchronous Response Generation
           fs.readFile(config.dataPath + key, 'utf8', function(err, value) {
               if (err) {
                   // Return File Not Found if file hasn't yet been created
                   response.writeHead(404, {'Content-Type': 'text/plain'});
                   response.end("The file (" + config.dataPath + key + ") does not yet exist.");
               } else {
                   // If the file exists, read it and return the sorted contents
                   var sorted = value.split(config.sortSplitString).sort().join('');
                   response.writeHead(200, {'Content-Type': 'text/plain'});
                   response.end(sorted);
               }
           });
           break;
       case 'POST':  // Synchronously append POSTed data to a file
           var postData = '';
           request
               .on('data', function (data) {
                   postData += data;
               })
               .on('end', function () {
                   fs.appendFile(config.dataPath + key, postData, function(err) {
                       if (err) {
                           //  Return error if unable to create/append to the file
                           response.writeHead(400, {'Content-Type': 'text/plain'});
                           response.end('Error: Unable to write file: ' + err);
                       } else {
                           //  Write or append posted data to a file, return "success" response
                           response.writeHead(200, {'Content-Type': 'text/plain'});
                           response.end('success');
                       }
                   });
               });
           break;
       default:
           response.writeHead(400, {'Content-Type': 'text/plain'});
           response.end("Error: Bad HTTP method: " + request.method);
   }
}).listen(config.serverPort);


console.log('synchronous server is running: ', config.serverPort);


四.总结
这篇博文是个人初次尝试NodeJS的一个小总结,如有写的不好还望大家多多的包含和指正。对于程序员来说,需要做的就是一直不停的学习,无论是否是自己主要从事的语言,对于学习多种语言,可以更加有助我们了解编程,对于一个开发者来说,最终的就是思想,因为语言的特性和框架的应用,一个熟练的编程者学习起来并不是难事,难就难在我们对于语言和框架的设计理念的理解。


来源:伯乐在线专栏作者 - 彭泽
cnblogs.com/pengze0902/p/6292751.html

0人打赏
您需要登录后才可以回帖
发表回复
极贡献
技术问答
专题荟萃
程序人生
视觉设计
Android开发
iOS开发
编程语言
前端开发
后端开发
服务器架构
软件测试
运维方案
创业路上



最热文章墙

  • 80581/379   【精品推荐】200多种Android动画效果的强悍框架,太全了,不看这个,再有动画的问题,不理你了^@^

  • 45809/191   情人节福利,程序员表白的正确姿势:改几行代码就变成自己的表白了

  • 45498/0   Python爬虫:常用浏览器的useragent

  • 42016/261   【精品推荐】Android版产品级的音乐播放器源码,功能太强大了,最好的产品原型有木有?

  • 39011/145   省时省力的Android组件群来了,非常棒的原型参考

  • 30465/143   2016抢红包软件及源码

  • 29524/71   原创表白APP,以程序员的姿势备战新年后的7夕,持续完善中!

  • 29452/2   超全!整理常用的iOS第三方资源

  • 24784/161   Android版类似UC浏览器:非常赞,产品级的源码

  • 23028/31   麻省理工的一帮疯子,真的实现了随意操控万物!(绝对黑科技)

  • 22730/26   Android工程师面试题大全

  • 22689/27   2016程序员跳槽全攻略

  • 22061/10   GitHub上排名前50的iOS项目:总有一款你用得着

  • 21063/21   码魂:程序员的牛B漫画

  • 19280/74   【持续更新中】Android福利贴(二):资料源码大放送

  • 19279/85   Android小而全的博客源码:非常适合全面掌握开发技巧

  • 19145/43   一个绚丽的loading动效分析与实现!

  • 19102/10   2016年最全的Android面试考题+答案 精编版

  • 18864/3   吐槽那些程序员的搞笑牛逼注释

  • 17848/45   惊艳的App引导页:背景图片切换加各个页面动画效果

  • 17748/104   Android带弹幕的视频播放器源码,来自大名鼎鼎的Bilibili弹幕网站

  • 17731/1   iOS 动画总结

  • 17709/82   仿京东商城客户端Android最新版,不错的原型和学习资料

  • 17363/25   个人收集的Android 各类功能源代码

  • 16888/5   新一代Android渠道打包工具:1000个渠道包只需要5秒

  • 16845/22   Android福利第三波【Android电子书】

  • 16652/1   iOS中文版资源库,非常全

  • 16648/81   【精品推荐】类似360安全卫士安Android源码:非常赞的产品原型

  • 16614/54   基于瀑布流的美女图片浏览App,有注释的源代码

  • 16488/18   用JavaScript 来开发iOS和Android 原生应用:React Native开源框架中文版来啦

  • 16482/10   女程序员的梦,众网友的神回复

  • 16187/11   年会上现场review代码是怎么样的体验!

  • 16185/23   珍藏多年的素材,灵感搜寻网站

  • 15791/19   65条最常用正则表达式,你要的都在这里了

  • 14980/16   基于Android支付宝支付设计和开发方案

  • 14391/11   有木有这样一张酷图帮你集齐所有git命令超实用

  • 14312/18   什么是真正的黑客:收获12200+Stars,人气远超微软开源VS

  • 14130/47   在线音乐播放器完整版(商用级的源码):非常赞,可听免费高品质专辑

  • 13929/62   【技巧一】搭配Android Studio,如何实现App远程真机debug?

  • 13814/0   GitHub iOS 库和框架Top100 

  • 13656/7   用程序员的姿势抢过年的火车票

  • 13625/7   一张图搞定iOS学习路线,非常全面

  • 13260/10   成为Java顶尖程序员 ,看这11本书就够了

  • 13219/10   微信支付终于成功了(安卓,iOS),在此分享

  • 13102/18   一张图搞定Android学习路线,非常全面

  • 12954/29   【持续更新中】Android福利贴(一):资料源码

  • 12857/4   基于Node.js的强大爬虫,能直接发布抓取的文章哦

  • 12661/4   46 个非常有用的 PHP 代码片段

  • 12129/3   即时通信第三方库

  • 11556/9   流媒体视频直播方案

  • 11512/18   八个最优秀的Android Studio插件

  • 11364/9   B站建开源工作组:APP想支持炫酷弹幕的看过来

  • 11191/2   【精品推荐】高质量PHP代码的50个实用技巧:非常值得收藏

  • 11136/9   烧了5亿美金,这家神秘的公司即将颠覆人类未来!

  • 11063/12   中国黑客的隐秘江湖:攻守对立,顶尖高手月入千万美元

  • 10495/1   基于node-webkit跨平台应用案例集之(一)

  • 10315/6   开箱即用!Android四款系统架构工具

  • 10113/11   十大技巧快速提升Android应用开发性能

  • 10098/4   10款GitHub上最火爆的国产开源项目——可以媲美西半球

  • 10055/2   Android性能优化视频,文档以及工具

  • 返回顶部