废话:自高中以来一直对编程充满激情,磨剑五年,如今要毕业了,我不想用我已经擅长的知识敷衍,而想以一个全新的领域去面向我的毕设——是时候学习一下node.js

node.js基础

对于JavaScript ES6的基本语法我就直接跳过了,我们说说node端的js与web端的js的不同,node与php的区别。

node完全的单线程

php语言本身为单线程,但是在响应用户的http请求时,借助apache可以生成多个线程,即每一个请求都是一个线程,线程之间是不共享变量的。

js语言本身也和php一样是单线程,与php作为web后端不同的是,node本身也是单线程响应用户请求的,这就意味着在开发nodejs时必须十分小心变量的共享产生错误。

将js进行到底:node学习笔记1-LMLPHP

将js进行到底:node学习笔记1-LMLPHP

node非阻塞与传统阻塞

传统的php是按照事先设定的脚本顺序执行代码的,若产生IO读写时,则会产生阻塞这次请求的线程,等待IO读写完成,继续顺序执行下面的代码,这个过程中,该线程不执行任何操作。

node是事件驱动的编程方式,这也是js的语言特性之一,即事件轮询:执行程序前注册好代码中描述的所有事件,cpu不停的轮着询问,“有没有事件现在要干啊?有没有?”,当到达事件执行时候就触发执行某些函数方法。需要搭配非阻塞IO,如果产生IO读写,依然保持轮询,只要有其他方法到达执行时刻就执行该事件,而不等待IO(前提也要保证下一个事件不需要这个IO获取的数据,否则再换一个事情做)。而IO读写完成也会触发一个事件,cpu会轮询到这个事件并继续执行。这就有点类似于操作系统课程里的:“多任务调度”。

在web开发中,阻塞的就是数据库操作了。

node有没有并发处理?

node.js最令人热衷讨论的就是他所谓的“高并发”,一台普通的笔记本电脑,用node书写普通的服务器能每秒处理上千个请求!但注意node是单线程处理请求的,事实上的并行运算是不存在的,每次只接受一个请求。所以高并发是真,但并行则是假象,node如何做到这样的高并发。这就是依赖于V8对js堆栈的高速处理,由于每一个请求处理速度极快,并且最重要是不等待请求产生的IO(上面提到node不会等待IO,这期间执行其他请求和事件),使之产生了某一刻响应上千请求的“假象”。

node里的JavaScript

node中的js也遵守ES标准,但是提供了与浏览器端一些不同的API,需要注意。

process

浏览器中js我们已经非常熟悉,他的根对象叫做window,这也是浏览器已经提供给开发者最大的API,比如document就是对象下的一个属性,alert就是对象下的一个方法。

node的根对象是process,这和window是一个道理。process是一个全局访问的对象。

模块——引入外部“类库”

c语言有include,java有import,php有require,几乎任何当下流行的语言都不可缺少引入类,包,或者模块的功能函数。在nodejs中,使用require取得外部模块,以扩展当前模块的功能。

绝对模块:在nodejs中,绝对模块指的是项目目录下的node_modules目录,这个目录将会在npm安装模块后自动生成,直接require模块名即可,例如 require('colors')。

【注:有些模块(例如fs)是node自带模块,类似于c语言stdio那样的标准库不需要用npm安装,啥是npm,这是一个nodejs的包管理器,类似于ubuntu下的apt-get或者gradle这类工具】

相对模块:一般是自己写的模块,是一些js文件,在require的时候需要输入路径引用,诸如:require(./xxx),无需后缀。

模块API的暴露问题:“返回对象内容或者不”

当你写了一个模块,这就类似与java 的一个外部包,但是js的模块更像是一个类,这个“类”(模块)提供了一些属性和方法,你可以考虑“暴露”一些内容,或者私有化他们,也就是让外部模块返回一个对象给当前的的模块使用,暴不暴露需要根据实际情况而定。看以下两个例子:

如何实现“暴露”?

//module.js
exports.name = 'john';
exports.data = 'this is some data';
var private = 1; exports.getPrivate = function(){
return private;
}
//index.js
var s = require("./module");
console.log(s.name);
console.log(s.data);
console.log(s.getPrivate());

将js进行到底:node学习笔记1-LMLPHP

关于不需要暴露对象内容的模块,比如colors模块,这些模块直接修改了js某些数据类型的prototype,为我们使用这些类型的数据对象提供了额外的功能:

require("colors");	//注意require不需要赋值给变量了
console.log("sssssssss".rainbow);

将js进行到底:node学习笔记1-LMLPHP

所以将模块需要暴露的东西放到module.exports对象即可,当然你还可以直接修改node的这个exports对象来暴露API。

事件

前端编程使用addEventListener为某个dom添加事件驱动。通过jQuery也可以封装为.on(),.live()等方法来注册事件。事件注册好后,根据注册要求,比如click,tap,当这些事情发生后就会触发事件注册的代码,这类似于数据库里的触发器。

在nodejs中可以直接使用on来注册事件:

//例如以下形式:
http.server(function(req,res){
var buf = "";
req.on("data",function(data){
buf += data;
});
req.on("end",function(){
console.log("数据接受完毕!");
});
});

这是一个http数据提交请求事件注册,共注册2个事件,第一个是数据到达事件,第二个是请求结束事件,根据node的“事件驱动”非阻塞的特性,http产生的以太网数据不能一下处理完成,node会让js执行别的请求或事件,当数据到达才回来执行第一个事件的函数,当数据结束后才执行第二个函数。

使用req.emit()可以解除这个事件绑定,类似于removeEventListener。

04-03 19:03