node.js的作用、回调、同步异步代码、事件循环
http://www.nodeclass.com/articles/39274
一.node.js的作用
I/O的意义,(I/O是输入/输出的简写,如:键盘敲入文本,输入,屏幕上看到文本显示输出。鼠标移动,在屏幕上看到鼠标的移动。终端的输入,和看到的输出。等等)
node.js想解决的问题,(处理输入,输入,高并发 。如 在线游戏中可能会有上百万个游戏者,则有上百万的输入等等)(node.js适合的范畴:当应用程序需要在网络上发送和接收数据时Node.js最为适合。这可能是第三方的API,联网设备或者浏览器与服务器之间的实时通信)
并发的意义,(并发这个术语描述的是事情会在同时发生并可能相互交互。Node的事件化的I/O模型让我们无需担心互锁和并发这两个在多线程异步I/O中常见的问题)
演示网络I/O
Js代码
- var http = require('http'),
- urls = ['www.baidu.com','www.10jqka.com.cn','www.duokan.com'];
- function fetchPage(url){
- var start = new Date();
- http.get({host:url},function(res){
- console.log("Got response from:" + url);
- console.log("Request took:",new Date() - start, "ms");
- });
- }
- for(var i=0; i<urls.length; i++){
- fetchPage(urls[i]);
- }
命名为,node.js
我们在终端里面运行node node.js
输出:

我们要求node.js访问三个url并报告收到响应的情况以及所耗费的时间。
我们可以看到两次输出的时间是不一样的。受各种影响,解析DNS请求的时间,服务器繁忙程序等等。
为什么javascript是一个事件驱动的语言
javascript围绕着最初与文档对象模型(DOM)相关的事件架构。开发人员可以在事件发生时做事情。这些事件有用户点击一个元素,页面完成加载等。使用事件,开发人员可以编写事件的监听器,当事件发生时被触发。
二.回调(Callback)
1.什么是回调
2.剖析回调
回调指的是将一个函数作为参数传递给另一个函数,并且通常在第一个函数完成后被调用。
例子:如jquery中的hide()方法,
Js代码
- 1,$("p").hide('slow');
- 2,$("p").hide('slow',function(){alert("The paragraph is now hidden")});
回调是可选的
1.就不需要回调
2.是有回调的,当段落隐藏完成后它就会被调用,显示一个alert提示。
为了可以看到带与不带回调的代码之间的区别
Js代码
- $("p").hide('slow');
- alert("The paragraph is now hidden");//1
- $("p").hide('slow',function(){alert("The paragraph is now hidden")});//2
1.是没有回调,执行顺序是一样但是,我们可以看到p段落还没有隐藏完全,alert就出来
2.是有回调的,执行则是hide完成后在alert
剖析回调
Js代码
- function haveBreakfast(food,drink,callback){
- console.log('Having barakfast of' + food + ', '+ drink);
- if(callback && typeof(callback) === "function"){
- callback();
- }
- }
- haveBreakfast('foast','coffee',function(){
- console.log('Finished breakfast. Time to go to work!');
- });
输出:
- Having barakfast of foast,coffee
- Finished breakfast. Time to go to work!
这里是创建了一个函数,有三个参数,第三个参数是callback,这个参数必须是个函数。
haveBreakfast函数将所吃的东西记录到控制台中然后调用作为参数传递给它的回调函数。
Node.js如何使用回调
node.js中使用filesystem模块从磁盘上读入文件内容的示例
Js代码
- var fs = require('fs');
- fs.readFile('somefile.txt','utf8',function(err,data){
- if(err) throw err;
- console.log(data);
- });
结果是:somefile.txt里面的内容。
1,fs(filesystem)模块被请求,以便在脚本中使用
2,讲文件系统上的文件路径作为第一个参数提供给fs.readFile方法
3,第二个参数是utf8,表示文件的编码
4,将回调函数作为第三个参数提供给fs.readFile方法
5,回调函数的第一个参数是err,用于保存在读取文件时返回的错误
6,回调函数的第二参数是data,用户保存读取文件所返回的数据
7,一旦文件被读取,回调就会被调用
8,如果err为真,那么就会抛出错误
9,如果err为假,那么来自文件的数据就可以使用
10,在本例中,数据会记录到控制台上
再一个,http模块,http模块使得开发人员可以创建http客户端和服务器。
Js代码
- var http = require('http');
- http.get({host:'shapeshed.com'},function(res){
- console.log("Got response:" + res.statusCode);
- }).on('error',function(e){
- console.log("Got error:" + e.message);
- });
结果:Got response:200
1,请求http模块,以便在脚本中使用
2,给http.get()方法提供两个参数
3,第一个参数是选项对象。在本示例中,要求获取shapeshed.com的主页
4,第二个参数是一个以响应作为参数的回调函数
5,当远程服务器返回相应时,会触发回调函数
6,在回调函数内记录响应状态码,如果有错误的话可以记录下来
接下来,我们看看有4个不同的I/O操作都在发生,他们都使用回调
Js代码
- var fs = require('fs'),
- http = require('http');
- http.get({host:'www.baidu.com'},function(res){
- console.log("baidu.com");
- }).on('error',function(e){
- console.log("Got error:" + e.message);
- });
- fs.readFile('somefile.txt','utf8',function(err,data){
- if(err) throw err;
- console.log("somefile");
- });
- http.get({host:'www.duokan.com'},function(res){
- console.log("duokan.com");
- }).on('error',function(e){
- console.log("Got error:" + e.message);
- });
- fs.readFile('somefile2.txt','utf8',function(err,data){
- if(err) throw err;
- console.log("somefile2");
- });
我们能知道哪个操作先返回吗?
猜测就是从磁盘上读取的两个文件先返回,因为无需进入网络,但是我们很难说哪个文件先返回,因为我们不知道文件的大小。对于两个主页的获取,脚本要进入网络,而响应时间则依赖于许多难以预测的事情,Node.js进程在还有已经注册的回调尚未触发之前将不会退出。回调首先解决不可预测性的方法,他也是处理并发(或者说一次做超过一件事情)的高效方法。
下面是我执行的结果

三.同步和异步代码
先看代码,同步(或者阻塞)代码
Js代码
- function sleep(milliseconds){
- var start = new Date().getTime();
- while((new Date().getTime() -start) < milliseconds){
- }
- }
- function fetchPage(){
- console.log('fetching page');
- sleep(2000);
- console.log('data returned from requesting page');
- }
- function fetchApi(){
- console.log('fetching api');
- sleep(2000);
- console.log('data returned from the api');
- }
- fetchPage();
- fetchApi();
当脚本运行时,fetchPage()函数会被调用,直到它返回之前,脚本的运行是被阻塞的,在fetchPage()函数返回之前,程序是不能移到fetchApi()函数中的。这称为阻塞操作。
Node.js几乎从不使用这种编码风格,而是异步地调用回调。
看下下面编码,,
Js代码
- var http = require('http');
- function fetchPage(){
- console.log('fetching page');
- http.get({host:'www.baidu.com',path:'/?delay=2000'},
- function(res){
- console.log('data returned from requesting page');
- }).on('error',function(e){
- console.log("There was an error" + e);
- });
- }
- function fetchApi(){
- console.log('fetching api');
- http.get({host:'www.baidu.com',path:'/?delay=2000'},
- function(res){
- console.log('data returned from requesting api');
- }).on('error',function(e){
- console.log("There was an error" + e);
- });
- }
- fetchPage();
- fetchApi();
允许这段代码的时候,就不再等待fetchPage()函数返回了,fetchApi()函数随之立刻被调用。代码通过使用回调,是非阻塞的了。一旦调用了,两个函数都会侦听远程服务器的返回,并以此触发回调函数。
注意这些函数的返回顺序是无法保证的,而是和网络有关。
四.事件循环
Node.js使用javascript的事件循环来支持它所推崇的异步编程风格。基本上,事件循环使得系统可以将回调函数先保存起来,而后当事件在将来发生时再运行。这可以是数据库返回数据,也可以是HTTP请求返回数据。因为回调函数的执行被推迟到事件反生之后,于是就无需停止执行,控制流可以返回到Node运行时的环境,从而让其他事情发生。
Node.js经常被当作是一个网络编程框架,因为它的设计旨在处理网络中数据流的不确定性。促成这样的设计的是事件循环和对回调的使用,他们似的程序员可以编写对网络或I/O事件进行响应的异步代码。
需要遵循的规则有:函数必须快速返回,函数不得阻塞,长时间运行的操作必须移到另一个进程中。
Node.js所不适合的地方包括处理大量数据或者长时间运行计算等。Node.js旨在网络中推送数据并瞬间完成。
node.js的作用、回调、同步异步代码、事件循环的更多相关文章
- node.js中的回调
同步和阻塞:这两个术语可以互换使用,指的是代码的执行会在函数返回之前停止.如果某个操作阻塞,那么脚本就无法继续,这意味着必须等待. 异步和非阻塞:这两个术语可以互换使用,指的是基于回调的.允许脚本并行 ...
- Node.js精进(2)——异步编程
虽然 Node.js 是单线程的,但是在融合了libuv后,使其有能力非常简单地就构建出高性能和可扩展的网络应用程序. 下图是 Node.js 的简单架构图,基于 V8 和 libuv,其中 Node ...
- node.js如何使用回调
Node.js到处使用回调,尤其在有I/O(输入/输出)操作的地方. 下面是在一个Node.js中使用filesystem模块中从磁盘上读入文件内容示例一: var fs = require('fs' ...
- Node.js NPM 作用
章节 Node.js NPM 介绍 Node.js NPM 作用 Node.js NPM 包(Package) Node.js NPM 管理包 Node.js NPM Package.json NPM ...
- Node.js标准的回调函数
Node.js标准的回调函数:第一个参数代表错误信息,第二个参数代表结果. function (err, data) 当正常读取时,err参数为null,data参数为读取到的String.当读取发生 ...
- 【Node100In1】01.去异步,解决掉Node.js万恶的回调陷阱
Node.js是基于事件驱动编程.异步函数随处可见,其中不乏一些常用库的方法.本例就以js中最常见的setTimeout的为例,试图改善一下回调的书写. 先来看一段伪代码: 我们实现一个需求,每隔一段 ...
- node.js 使用domain模块捕获异步回调中的异常
和其他服务器端语言相比,貌似node.js 对于异常捕捉确实非常困难. 首先你会想到try/catch ,但是在使用过程中我们会发现并没有真正将错误控制在try/catch 语句中. 为什么? 答案是 ...
- 《深入浅出Node.js》第4章 异步编程
@by Ruth92(转载请注明出处) 第4章 异步编程 Node 能够迅速成功并流行起来的原因: V8 和 异步 I/O 在性能上带来的提升: 前后端 JavaScript 编程风格一致 一.函数式 ...
- 《深入浅出Node.js》第3章 异步I/O
@by Ruth92(转载请注明出处) 第3章 异步I/O Node 的基调:异步 I/O.事件驱动.单线程. Node 不再是一个服务器,而是一个可以基于它构建各种高速.可伸缩网络应用的平台. No ...
随机推荐
- 搭建 hexo,在执行 hexo deploy 后,出现 error deployer not found:github 的错误
hexo 更新到3.0之后,deploy的type 的github需要改成git 改了之后执行npm install hexo-deployer-git --save 然后再部署试试 官网说明: ht ...
- CentOS6 Squid代理服务器的安装与配置
代理服务器英文全称是Proxy Server,其功能就是代理网络用户去取得网络信息.Squid是一个缓存Internet 数据的软件,其接收用户的下载申请,并自动处理所下载的数据.当一个用户想要下载一 ...
- Java 性能优化实战记录(2)---句柄泄漏和监控
前言: Java不存在内存泄漏, 但存在过期引用以及资源泄漏. (个人看法, 请大牛指正) 这边对文件句柄泄漏的场景进行下模拟, 并对此做下简单的分析.如下代码为模拟一个服务进程, 忽略了句柄关闭, ...
- kafka集群和zookeeper集群的部署,kafka的java代码示例
来自:http://doc.okbase.net/QING____/archive/19447.html 也可参考: http://blog.csdn.net/21aspnet/article/det ...
- hdu3033 I love sneakers! 分组背包变形
分组背包要求每一组里面只能选一个,这个题目要求每一组里面至少选一个物品. dp[i, j] 表示前 i 组里面在每组至少放进一个物品的情况下,当花费 j 的时候,所得到的的最大价值.这个状态可以由三个 ...
- C#笔记 -----扩展方法
在我们使用vs自带的工具函数时,如: string str='111';str.toInt(); 有没有想到过他们是怎么来的? 这就是C# 的 方法扩展: age: using system: pu ...
- POJ-1741 Tree (树上点分治)
题目大意:一棵带边权无根树,边权代表距离,求距离小于等于k的点对儿数. 题目分析:这两个点之间的路径只有两种可能,要么经过根节点,要么在一棵子树内.定义depth(i)表示点 i 到根节点的距离,be ...
- poj1502 最短路
题意:有n个处理器,给出n*n的邻接矩阵的一半,表示相互之间传输信息的花费(另一半与给出的一半对称,即双向都可传输),x表示不能传输,问从第一个处理器传输到所有处理器的最小花费总和是多少. 就是跑一遍 ...
- jQuery判断元素是否在可视区
假设此元素为 #item,先说几个关键的属性: $('#item').offset().top#item 的绝对偏移量,指#item的实际尺寸(即不包括外边框margin)的上边界到页面顶端的距离.这 ...
- spring源码学习之:xml配置文件标签自定义
Spring框架从2.0版本开始,提供了基于Schema风格的XML扩展机制,允许开发者扩展最基本的spring配置文件(一 般是classpath下的spring.xml).试想一下,如果我们直接在 ...