什么是Node.js?css
它是一个专一于实现高性能Web服务器优化的专家,几经探索,几经挫折后,遇到V8而诞生的项目。Node.js是一个让JavaScript运行在服务器端的开发平台,它让JavaScript的触角伸到了服务器端,能够与PHP、JSP、Python、Ruby分庭抗礼。但Node彷佛有点不一样:Node.js不是一种独立的语言,与PHP、JSP、Python、Perl、Ruby的“既是语言,也是平台”不一样,Node.js的使用JavaScript进行编程,运行在JavaScript引擎上(V8)。与PHP、JSP等相比(PHP、JSP、.net都须要运行在服务器程序上,Apache、Naginx、Tomcat、IIS。),Node.js跳过了Apache、Naginx、IIS等HTTP服务器,它本身不用建设在任何服务器软件之上。Node.js的许多设计理念与经典架构(LAMP = Linux + Apache + MySQL + PHP)有着很大的不一样,能够提供强大的伸缩能力。Node.js没有web容器。Node.js自身哲学,是花最小的硬件成本,追求更高的并发,更高的处理性能。html
学习Node.js须要了解:node
单线程:在Java、PHP或者.net等服务器端语言中,会为每个客户端链接建立一个新的线程,而每一个线程须要耗费大约2MB内存,理论上一个8GB内存的服务器能够同时链接的最大用户数为4000个左右。要让Web应用程序支持更多的用户,就须要增长服务器的数量,而Web应用程序的硬件成本固然就上升了。Node.js不为每一个客户链接建立一个新的线程,而仅仅使用一个线程。当有用户链接了,就触发一个内部事件,经过非阻塞I/O、事件驱动机制,让Node.js程序宏观上也是并行的。使用Node.js,一个8GB内存的服务器,能够同时处理超过4万用户的链接。另外,单线程的带来的好处,还有操做系统彻底再也不有线程建立、销毁的时间开销。程序员
非阻塞I/O:由于CPU的效率是远远高于I/O设备的执行效率的,若是让CPU去等待I/O的执行,将会极大地浪费时间,下降性能,好比在访问数据库或者读文件的时候,在传统的单线程处理机制中,在执行了访问数据库或文件代码以后,整个线程都将暂停下来(阻塞I/O),等待数据库或者文件系统返回结果才能执行后面的代码。I/O阻塞了代码的执行极大地下降了程序的执行效率。因为Node.js中采用了非阻塞型I/O机制,所以在执行了访问数据库或文件的代码以后,将当即转而执行其余代码,把返回结果的处理代码放在回调函数中,从而提升了程序的执行效率。当某个I/O执行完毕时,将以事件的形式通知执行I/O操做的线程,线程执行这个事件的回调函数。为了处理异步I/O,线程必须有事件循环,不断的检查有没有未处理的事件,依次予以处理。阻塞模式下,一个线程只能处理一项任务,要想提升吞吐量必须经过多线程。而非阻塞模式下,一个线程永远在执行计算操做,这个线程的CPU核心利用率永远是100%。因此,这是一种特别有哲理的解决方案:与其人多,可是好多人闲着;还不如一我的玩命,往死里干活儿。web
事件驱动:在Node中,客户端请求创建链接,提交数据等行为,会触发相应的事件。在Node中,在一个时刻,只能执行一个事件回调函数,可是在执行一个事件回调函数的中途,能够转而处理其余事件,而后返回继续执行原事件的回调函数,这种处理机制,称为“事件环”机制。Node.js底层是C++(V8也是C++写的),底层代码中,近半数都用于事件队列、回调函数队列的构建。数据库
优缺点:由于单线程,在处理大规模并发的任务中仍是会显得力不从心的,好比在CPU密集型事务中就会遇到瓶颈,另外就是node.js是没有web容器的,代码直接没有根目录的说法,在必定程度上为程序员增长了代码量,但也提升了灵活性,为高级路由带来了极大的方便,在node.js中回调函数会有很深的层次,为代码的阅读多多少少形成了必定的障碍。善于处理异步事件(callback),处理同步事务中须要额外的负担。编程
适用范围:当应用程序须要处理大量并发的I/O,而在向客户端发出响应以前,应用程序内部并不须要进行很是复杂的处理的时候,Node.js很是适合。Node.js也很是适合与web socket配合,开发长链接的实时交互应用程序。好比:用户表单收集、考试系统、聊天室、图文直播、提供JSON的API(为前台Angular使用)。服务器
应用举例:多线程
1 //require表示引包,引包就是引用本身的一个特殊功能 2 var http = require("http"); 3 var fs = require("fs"); 4 5 //建立服务器,参数是一个回调函数,表示若是有请求进来,要作什么 6 var server = http.createServer(function(req,res){ 7 if(req.url == "/fang"){ 8 fs.readFile("./test/xixi.html",function(err,data){ 9 //req表示请求,request; res表示响应,response 10 //设置HTTP头部,状态码是200,文件类型是html,字符集是utf8 11 res.writeHead(200,{"Content-type":"text/html;charset=UTF-8"}); 12 res.end(data); 13 }); 14 }else if(req.url == "/yuan"){ 15 fs.readFile("./test/haha.html",function(err,data){ 16 //req表示请求,request; res表示响应,response 17 //设置HTTP头部,状态码是200,文件类型是html,字符集是utf8 18 res.writeHead(200,{"Content-type":"text/html;charset=UTF-8"}); 19 res.end(data); 20 }); 21 }else if(req.url == "/1.jpg"){ 22 fs.readFile("./test/0.jpg",function(err,data){ 23 //req表示请求,request; res表示响应,response 24 //设置HTTP头部,状态码是200,文件类型是html,字符集是utf8 25 res.writeHead(200,{"Content-type":"image/jpg"}); 26 res.end(data); 27 }); 28 }else if(req.url == "/bbbbbb.css"){ 29 fs.readFile("./test/aaaaaa.css",function(err,data){ 30 //req表示请求,request; res表示响应,response 31 //设置HTTP头部,状态码是200,文件类型是html,字符集是utf8 32 res.writeHead(200,{"Content-type":"text/css"}); 33 res.end(data); 34 }); 35 }else{ 36 res.writeHead(404,{"Content-type":"text/html;charset=UTF-8"}); 37 res.end("没有这个页面呦"); 38 } 39 }); 40 41 //运行服务器,监听3000端口(端口号能够任改) 42 server.listen(3000);
上面的代码就是说明这样的一种状况,当用户输入了URL以后进行处理的时候,node.js能够对URL进行解析而且按照本身设定好的路径来根据相应的字段找到多是有着好几个文件夹下的一个文件,这个文件的文件名可能与url中的字段彻底不一样,也就是说能够把字段看作一个地址(指针),指向相应的文件所在的物理位置!架构
if(req.url == "/fang"){ fs.readFile("./test/xixi.html",function(err,data){ //req表示请求,request; res表示响应,response //设置HTTP头部,状态码是200,文件类型是html,字符集是utf8 res.writeHead(200,{"Content-type":"text/html;charset=UTF-8"}); res.end(data); }); }
好比上面的代码,明明URL是IP+/fang,但是在返回数据的时候将当前目录下的test文件夹下的xixi.html文件做为返回对象。为网站的路由设计提供了极大的便利。