Afin de comprendre le fonctionnement de Node.js, vous devez d'abord Comprenez certaines des fonctionnalités clés qui rendent Javascript adapté au développement côté serveur. Javascript est un langage simple mais flexible, et cette flexibilité lui permet de résister à l'épreuve du temps. Des fonctionnalités telles que des fonctions et des fermetures font de Javascript un langage idéal pour le développement Web.
Il existe un préjugé selon lequel Javascript n'est pas fiable, mais ce n'est pas le cas. Les préjugés contre Javascript proviennent du DOM. DOM est une API fournie par les fabricants de navigateurs pour que Javascript puisse interagir avec les navigateurs. Il existe des différences dans le DOM implémenté par les différents fabricants de navigateurs. Cependant, Javascript lui-même est un langage bien défini qui peut s'exécuter dans différents navigateurs et Node.js. Dans cette section, je présenterai d'abord quelques bases de Javascript et comment Node.js utilise Javascript pour fournir une plateforme de développement Web avec d'excellentes performances.
Javascript utilise le mot-clé var
pour définir les variables. Par exemple, le code suivant crée une variable nommée foo
et l'affiche sur la ligne de commande. (Vous pouvez exécuter le fichier de code suivant dans la ligne de commande via node variable.js
.)
var foo = 123;console.log(foo); // 123
Environnement d'exécution Javascript (navigateur ou Node.js) définit généralement certaines variables globales que nous pouvons utiliser, comme un objet console
. L'objet console
contient une fonction membre log
. La fonction log
peut accepter n'importe quel nombre de paramètres et les afficher. Nous rencontrerons ensuite des objets plus globaux et vous constaterez que Javascript possède la plupart des fonctionnalités qu'un bon langage de programmation devrait contenir.
Javascript prend en charge les opérateurs arithmétiques courants (+
, -
, *
, /
, %
). Par exemple, le code suivant :
var foo = 3; var bar = 5; console.log(foo+1); //4 console.log(foo / bar); //0.6 console.log(foo * bar); //15 console.log(foo - bar); //-2 console.log(foo % 2); //取余:1
Les valeurs booléennes incluent true
et false
. Vous pouvez attribuer à une variable une valeur de true
ou false
et y effectuer des opérations booléennes. Par exemple, le code suivant :
var foo = true; console.log(foo); //true//常见的布尔操作符号: &&,||, ! console.log(true && true); //true console.log(true && false); /false console.log(true || false); //true console.log(false || false); //false console.log(!true); //false console.log(!false); //true
En Javascript, on peut créer un tableau via []
. Les objets tableau contiennent de nombreuses fonctions utiles, comme le montre le code suivant :
var foo = []; foo.push(1); //添加到数组末尾 console.log(foo); // [1] foo.unshift(2); //添加到数组头部 console.log(foo); // [2, 1]//数组起始位置从0开始 console.log(foo[0]); // 2
Les littéraux d'objet {}
sont généralement utilisés en Javascript pour créer des objets, comme indiqué dans ce qui suit code :
var foo = {}; console.log(foo); // {} foo.bar = 123; console.log(foo); // {bar: 123}
Le code ci-dessus ajoute des propriétés d'objet au moment de l'exécution. Nous pouvons également définir des propriétés d'objet lors de la création d'un objet :
var foo = { bar: 123 }; console.log(foo); // {bar: 123}
Les littéraux d'objet peuvent être imbriqués dans d'autres littéraux d'objet. Quantité, par exemple, comme indiqué dans le code suivant :
var foo = { bar: 123, bas: { bas1: 'some string', bas2: 345 } }; console.log(foo);
Bien entendu, les littéraux d'objet peuvent également contenir des tableaux :
var foo = { bar: 123, bas: [1,2,3] }; console.log(foo);
Les tableaux peuvent également contenir des littéraux d'objet :
var foo = { bar: 123, bas: [{ qux: 1 }, { qux: 2 }, { qux: 3 }] }; console.log(foo.bar); //123 console.log(foo.bas[0].qux); // 1 console.log(foo.bas[2].qux); // 2
Les fonctions Javascript sont très puissantes Nous allons progressivement le comprendre à travers une série d'exemples.
La structure habituelle des fonctions Javascript est la suivante :
function functionName(){ //函数体 }
Toutes les fonctions Javascript ont des valeurs de retour. Sans instruction de retour explicite, la fonction renvoie undefined
. Par exemple, le code suivant est affiché :
function foo(){return 123;}console.log(foo); // 123function bar(){ }console.log(bar()); // undefined
Nous exécutons la fonction immédiatement après l'avoir définie, la mettons entre crochets ()
et appelons la fonction. Comme le montre le code suivant :
(function foo(){ console.log('foo was executed!'); })();
La raison pour laquelle la fonction d'exécution immédiate apparaît est pour créer une nouvelle portée de variable. if
, else
, while
ne créeront pas de nouvelle portée de variable, comme indiqué dans le code suivant :
var foo = 123;if(true){ var foo = 456; }console.log(foo); // 456
En Javascript, nous créons une nouvelle portée de variable via des fonctions, telles que Utiliser une fonction d'exécution immédiate :
var foo = 123;if(true){ (function(){ var foo = 456; })(); }console.log(foo); // 123
Dans le code ci-dessus, nous n'avons pas donné de nom à la fonction, qui est appelée fonction anonyme.
Une fonction sans nom est appelée fonction anonyme. En Javascript, nous pouvons attribuer des fonctions à des variables. Si nous prévoyons d'utiliser la fonction comme variable, nous n'avons pas besoin de nommer la fonction. Deux manières d'écriture équivalentes sont données ci-dessous :
var foo1 = function nameFunction(){ console.log('foo1'); } foo1(); // foo1var foo2 = function(){ console.log('foo2'); } foo2(); // foo2f
On dit que si un langage de programmation peut traiter les fonctions comme des variables, c'est un excellent langage de programmation, et Javascript y est parvenu.
Puisque Javascript nous permet d'attribuer des fonctions à des variables, nous pouvons transmettre des fonctions en tant que paramètres à d'autres fonctions. Les fonctions qui prennent des fonctions comme arguments sont appelées fonctions d'ordre supérieur. setTimeout
est une fonction courante d'ordre supérieur.
setTimeout(function(){console.log('2000 milliseconds have passed since this demo started'); }, 2000);
Si vous exécutez le code ci-dessus dans Node.js, vous verrez la fenêtre de commande afficher des informations après 2 secondes. Dans le code ci-dessus, nous avons passé une fonction anonyme comme premier paramètre de setTimeout
. Nous pouvons également transmettre une fonction normale :
function foo(){ console.log('2000 milliseconds have passed since this demo started'); } setTimeout(foo, 200);
Maintenant que nous avons découvert les littéraux et les fonctions d'objet, nous allons découvrir le concept de fermeture.
Une fermeture est une fonction qui peut accéder à des variables à l'intérieur d'autres fonctions. Si vous définissez une autre fonction à l'intérieur d'une fonction, la fonction interne peut accéder aux variables de la fonction externe. Il s'agit d'une forme courante de fermeture. Nous allons l'expliquer avec quelques exemples.
Dans le code ci-dessous, vous pouvez voir que la fonction interne est capable d'accéder aux variables de la fonction externe :
function outerFunction(arg){ var variableInOuterFunction = arg; function bar(){console.log(variableInOuterFunction); } bar(); } outerFunction('hello closure!'); // hello closure!
令人惊喜的是:内部函数在外部函数返回之后依然可以访问外部函数作用域中的变量。这是因为,变量仍然被绑定于内部函数,不依赖于外部函数。例如:
function outerFunction(arg){ var variableInOuterFunction = arg; return function(){console.log(variableInOuterFunction); } }var innerFunction = outerFunction('hello closure!'); innerFunction(); // hello closure!
现在,我们已经了解了闭包,接下来,我们会探究一下使Javascript成为一门适合服务器端编程的语言的原因。
Node.js致力于开发高性能应用程序。接下来的部分,我们会介绍大规模I/O问题,并分别展示传统方式及Node.js是如何解决这个问题的。
大多数Web应用通过硬盘或者网络(例如查询另一台机器的数据库)获取数据,从硬盘或网络获取数据的速度远远慢于CPU的处理周期。当收到一个HTTP请求以后,我们需要从数据库获取数据,请求会一直等待直到获取数据完成。这些创建的连接和还未结束的请求会消耗服务器的资源(内存和CPU)。为了使同一台Web服务器能够处理大规模请求,我们需要解决大规模I/O问题。
传统的Web服务器为每一个请求创建一个新的进程,这是一种对内存和CPU开销都很昂贵的操作。PHP最开始就是采用的这种方法。在等待响应期间,进程仍然会消耗资源,并且进程的创建更慢。所以现代Web应用大多使用线程池的方法。
现代Web服务器使用线程池来处理每个请求。线程和进程相比,更加轻量级。在创建线程池以后,我们就不再需要为开始或结束进程而付出额外代价。当收到一个请求,我们为它分配一个线程。然而,线程池仍然会浪费一些资源。
我们知道为请求分别创建进程或者线程会导致系统资源浪费。与之相对,Node.js采取了单线程来处理请求。单线程服务器的性能优于线程池服务器的理念并不是Node.js首创,Nginx也是基于这种理念。Nginx是一种单线程服务器,能够处理极大数量的并发请求。
Javascript是单线程的,如果你有一个耗时操作(例如网络请求),就必须使用回调。下面的代码使用setTimeout
模拟了一个耗时操作,可以用Node.js执行。
function longRunningOperation(callback){ setTimeout(callback, 3000); }function UserClicked(){ console.log('starting a long operation'); longRunningOperation(function(){ console.log('ending a long operation'); }) } UserClicked();
让我们模拟一下Web请求:
function longRunningOperation(callback){ setTimeout(callback, 3000); }function webRequest(request){ console.log('starting a long operation for request:', request.id); longRunningOperation(function(){console.log('ending a long operation for request:', request.id); }); } webRequest({id: 1}); webRequest({id: 2}); //输出 //starting a long operation for request: 1//starting a long operation for request: 2//ending a long operation for request: 1//ending a long operation for request: 2
Node.js的核心是一个event loop
。event loop
使得任何用户图形界面应用程序可以在任何操作系统中工作。当事件被触发时(例如:用户点击鼠标),操作系统调用程序的某个函数,程序执行函数中的代码。之后,程序准备响应已经在队列中的事件或尚未出现的事件。
通常,在GUI程序中,当由一个事件调用的函数执行期间,其它事件不会被处理。因此,当你在相关函数中执行耗时操作时,GUI会变得无响应。这种CPU资源的短缺被成为饥饿
。
Node.js基于和GUI应用程序相同的event loop
原则。因此,它也会面临饥饿的问题。为了帮助更好的理解,我们通过几个例子来说明:
console.time('timer'); setTimeout(function(){ console.timeEnd('timer'); //timer: 1002.615ms }, 1000)
运行这段代码,与我们期望的相同,终端显示的数字在1000ms左右。
接下来我们想写一段耗时更长的代码,例如一个未经优化的计算Fibonacci数列的方法:
console.time('timeit');function fibonacci(n){ if(n<2){return 1; }else{return fibonacci(n-2) + fibonacci(n-1); } } fibonacci(44);console.timeEnd('timeit'); //我的电脑耗时 11863.331ms,每台电脑会有差异
现在我们可以模拟Node.js的线程饥饿。setTimeout
用于在指定的时间以后调用函数,如果我们在函数调用以前,执行一个耗时方法,由于耗时方法占用CPU和Javascript线程,setTimeout
指定的函数无法被及时调用,只能等待耗时方法运行结束以后被调用。例如下面的代码:
function fibonacci(n){ if(n<2){return 1; }else{return fibonacci(n-2) + fibonacci(n-1); } }console.time('timer'); setTimeout(function(){ console.timeEnd('timer'); // 输出时间会大于 1000ms }, 1000) fibonacci(44);
所以,如果你面临CPU密集型场景,Node.js并不是最佳选择,但也很难找到其它合适的平台。但是Node.js非常适用于I/O密集型场景。
Node.js适用于I/O密集型。单线程机制意味着Node.js作为Web服务器会占用更少的内存,能够支持更多的请求。与执行代码相比,从数据库获取数据需要花费更多的时间。下图展示了传统的线程池模型的服务器是如何处理用户请求的:
Node.js服务器处理请求的方式如下图。因为所有的工作都在单线程内完成,所以消耗更少的内存,同时因为不需要切换线程,所以CPU负载更小。
Node.js中的所有Javascript通过V8 Javascript引擎执行。V8产生于谷歌Chrome项目,V8在Chrome中用于运行Javascript。V8不仅速度更快,而且很容易被集成到其它项目。
精通Javascript使得Node.js开发者不仅能够写出更加容易维护的项目,而且能够利用到Javascript生态链的优势。
Javascript变量的默认值是undefined
。如下列代码所示:
var foo;console.log(foo); //undefined
变量不存在的属性也会返回undefined
var foo = {bar: 123}; console.log(foo.bar); // 123 console.log(foo.bas); // undefined
需要注意Javascript当中 ==
与===
的区别。==
会对变量进行类型转换,===
不会。推荐的用法是总是使用===
。
console.log(5 == '5'); // true console.log(5 === '5'); // false
null
是一个特殊的Javascript对象,用于表示空对象。而undefined
用于表示变量不存在或未初始化。我们不需要给变量赋值为undefined
,因为undefined
是变量的默认值。
透露模块模式的关键在于Javascript对闭包的支持以及能够返回任意对象的能力。如下列代码所示:
function printableMessage(){ var message = 'hello'; function setMessage(newMessage){if(!newMessage) throw new Error('cannot set empty message'); message = newMessage; } function getMessage(){return message; } function printMessage(){ console.log(message); } return { setMessage: setMessage, getMessage: getMessage, printMessage: printMessage }; }var awesome1 = printableMessage(); awesome1.printMessage(); //hellovar awesome2 = printableMessage(); awesome2.setMessage('hi'); awesome2.printMessage(); // hi awesome1.printMessage(); //hello
this
this
总是指向调用函数的对象。例如:
var foo = { bar: 123, bas: function(){console.log('inside this.bar is: ', this.bar); } }console.log('foo.bar is:', foo.bar); //foo.bar is: 123 foo.bas(); //inside this.bar is: 123
由于函数bas
被foo
对象调用,所以this
指向foo
。如果是纯粹的函数调用,则this
指向全局变量。例如:
function foo(){ console.log('is this called from globals? : ', this === global); //true } foo();
如果我们在浏览器中执行上面的代码,全局变量global
会变为window
。
如果函数的调用对象改变,this
的指向也会改变:
var foo = { bar: 123 };function bas(){ if(this === global){console.log('called from global'); } if(this === foo){console.log('called from foo'); } }//指向global bas(); //called from global//指向foo foo.bas = bas; foo.bas(); //called from foo
如果通过new
操作符调用函数,函数内的this
会指向由new
创建的对象。
function foo(){ this.foo = 123; console.log('Is this global? : ', this == global); } foo(); // Is this global? : true console.log(global.foo); //123var newFoo = new foo(); //Is this glocal ? : false console.log(newFoo.foo); //123
通过上面代码,我们可以看到,在通过new
调用函数时,函数内的this
指向发生改变。
Javascript通过new
操作符及原型属性可以模仿面向对象的语言。每个Javascript对象都有一个被称为原型的内部链接指向其他对象。
当我们调用一个对象的属性,例如:foo.bar
,Javascript会检查foo
对象是否存在bar
属性,如果不存在,Javascript会检查bar
属性是否存在于foo._proto_
,以此类推,直到对象不存在_proto_
。如果在任何层级发现属性的值,则立即返回,否则,返回undefined
。
var foo ={}; foo._proto_.bar = 123; console.log(foo.bar); //123
当我们通过new
操作符创建对象时,对象的_proto_
会被赋值为函数的prototype
属性,例如:
function foo(){}; foo.prototype.bar = 123;var bas = new foo();console.log(bas._proto_ === foo.prototype); //trueconsole.log(bas.bar);
函数的所有实例共享相同的prototype
function foo(){}; foo.prototype.bar = 123; var bas = new foo(); var qux = new foo(); console.log(bas.bar); //123 console.log(qux.bar); //123 foo.prototype.bar = 456; console.log(bas.bar); //456 console.log(qux.bar); //456
只有当属性不存在时,才会访问原型,如果属性存在,则不会访问原型。
function foo(){}; foo.prototype.bar = 123;var bas = new foo();var qux = new foo(); bas.bar = 456;console.log(bas.bar);//456console.log(qux.bar); //123
上面的代码表明,如果修改了bas.bar
, bas._proto_.bar
就不再被访问。
Javascript的异常处理机制类似其它语言,通过throw
关键字抛出异常,通过catch
关键字捕获异常。例如:
try{ console.log('About to throw an error'); throw new Error('Error thrown'); } catch(e){ console.log('I will only execute if an error is thrown'); console.log('Error caught: ', e.message); }finally{ console.log('I will execute irrespective of an error thrown'); }
本章,我们介绍了一些Node.js及Javascript的重要概念,知道了Node.js适用于开发数据密集型应用程序。下章我们将开始介绍如何使用Node.js开发应用程序。
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!