Nodejs的单线程非阻塞I/O事件驱动
在 Java、PHP 或者.net 等服务器端语言中,会为每一个客户端连接创建一个新的线程。 而每个线程需要耗费大约 2MB 内存。也就是说,理论上,一个 8GB 内存的服务器可以同时 连接的最大用户数为 4000 个左右。要让 Web 应用程序支持更多的用户,就需要增加服务器 的数量,而 Web 应用程序的硬件成本当然就上升了。 Node.js 不为每个客户连接创建一个新的线程,而仅仅使用一个线程。当有用户连接了, 就触发一个内部事件,通过非阻塞 I/O、事件驱动机制,让 Node.js 程序宏观上也是并行的。 使用 Node.js,一个 8GB 内存的服务器,可以同时处理超过4 万用户的连接。
Nodejs 回调处理异步
//错误的写法:
function getData (){
// 模拟请求数据
var result='';
setTimeout(function(){
result='这是请求到的数据'
},200);
return result;
}
console.log( getData ()); /* 异步导致请求不到数据 */
// 正确的处理异步 :
function getData (callback){
// 模拟请求数据
var result='';
setTimeout (function(){
result='这是请求到的数据';
callback(result);
},200);
}
getData (function( data ){
console.log(data);
});
Nodejs events 模块处理异步
Node.js 有多个内置的事件,我们可以通过引入 events 模块,并通过实例化 EventEmitter 类来绑定和监听事件。
/*
Node.js 事件循环:
Node.js 是单进程单线程应用程序,但是通过事件和回调支持并发,所以性能非常高。
Node.js 的每一个 API 都是异步的,并作为一个独立线程运行,使用异步函数调用,并处理并发。
Node.js 有多个内置的事件,我们可以通过引入 events 模块,并通过实例化 EventEmitter 类来绑定和监听事件,
如下实例:
* */
// 引入 events 模块
var events=require('events');
//console.log(events);
var EventEmitter=new events.EventEmitter();
//广播 和接收广播
EventEmitter.on('to_mime',function(data){
console.log(data);
})
//监听to_parent的广播
EventEmitter.on('to_parent',function(data){
//console.log('接收到了这个广播事件');
console.log(data);
EventEmitter.emit('to_mime','给mime发送的数据')
})
setTimeout(function(){
console.log('开始广播...');
//广播to_parent事件
EventEmitter.emit('to_parent','发送的数据')
},1000);
我的上一篇博文:http://blog.csdn.net/cckevincyh/article/details/78637576
中读取mime.json文件,返回指定mime type,我们使用的是回调来处理异步。现在我们改造一下,通过events 模块来处理异步。
我们现在测试下我们的代码:
var fs=require('fs');
var events=require('events');
var EventEmitter=new events.EventEmitter();
function getMime(){
fs.readFile('mime.json',function(err,data){
EventEmitter.emit('data',data)
})
}
getMime();/*执行方法*/
//监听广播数据
EventEmitter.on('data',function(mime){
console.log(mime.toString());
})
好了,通过events 模块我们可以获取到我们的数据了,现在我们开始改造我们的代码:
exports.getMime=function(fs,EventEmitter,extname){ /*获取后缀名的方法*/
fs.readFile('./mime.json',function(err,data){
if(err){
console.log('mime.json文件不存在');
return false;
}
//console.log(data.toString());
var Mimes=JSON.parse(data.toString());
var result= Mimes[extname] || 'text/html';
EventEmitter.emit('to_mime',result);
})
}
//引入http模块
var http=require('http');
//fs模块
var fs=require('fs');
//path模块
var path=require('path'); /*nodejs自带的模块*/
//url模块
var url=require('url');
var events=require('events');
var EventEmitter=new events.EventEmitter();
var mimeModel=require('./model/getmimefromfile_events.js');
//引入扩展名的方法是在文件里面获取到的。
//console.log(mimeModel.getMime('.css')); //获取文件类型
http.createServer(function(req,res){
//http://localhost:8001/news.html /news.html
//http://localhost:8001/index.html /index.html
//css/dmb.bottom.css
//xxx.json?214214124
var pathname=url.parse(req.url).pathname;
console.log(pathname);
if(pathname=='/'){
pathname='/index.html'; /*默认加载的首页*/
}
//获取文件的后缀名
var extname=path.extname(pathname);
if(pathname!='/favicon.ico'){ /*过滤请求favicon.ico*/
//console.log(pathname);
//文件操作获取 static下面的index.html
fs.readFile('static/'+pathname,function(err,data){
if(err){ /*没有这个文件*/
console.log('404');
fs.readFile('static/404.html',function(error,data404){
if(error){
console.log(error);
}
res.writeHead(404,{"Content-Type":"text/html;charset='utf-8'"});
res.write(data404);
res.end(); /*结束响应*/
})
}else{ /*返回这个文件*/
mimeModel.getMime(fs,EventEmitter,extname); /*调用获取数据的方法*/
EventEmitter.on('to_mime',function(mime){
res.writeHead(200,{"Content-Type":""+mime+";charset='utf-8'"});
//res.write(data);
res.end(data); /*结束响应*/
})
}
})
}
}).listen(8002);
setMaxListeners():
在默认情况下,同一个指定的事件,最多可以绑定10个事件处理函数。当绑定数超出时,启动nodejs时,会有相应的错误提示。也可以通过下面的方法修改:
EventEmitter.setMaxListeners (n)
//给EventEmitter设置最大监听
//参数1: n 数字类型,最大监听数