极快的node.js:来自领英(LinkedIn)移动的10个性能技巧

发布时间:2018-01-16 22:22:35

在之前发布的文章中,讨论了我们是如何测试领英移动堆栈的,包括我们的Node.js移动服务。现在,我们将告诉你我们是如何让移动服务变得更快的。下面是我们使用Node.js时遵循的10个性能规则:


1. 避免使用同步代码


在设计上,Node.js是单线程的。为了能让一个单线程处理许多并发的请求,你可以永远不要让线程等待阻塞,同步或长时间运行的操作。Node.js的一个显著特征是:它从上到下的设计和实现都是为了实现异步。这让它非常适合用于事件型程序。

不幸的是,还是有可能会发生同步/阻塞的调用。例如,许多文件系统操作同时拥有同步和异步的版本,比如和。即使你用代码来控制同步方法,但还是有可能不注意地用到阻塞调用的外部函数库。当你这么做时,对性能的影响是极大的。

// Good: write files asynchronously

fs.writeFile('message.txt', 'Hello Node', function (err) {

console.log("It's saved and the server remains responsive!");

});

 

// BAD: write files synchronously

fs.writeFileSync('message.txt', 'Hello Node');

console.log("It's saved, but you just blocked ALL requests!");

我们的初始化log在实现时无意地包含了一个同步调用来将内容写入磁盘。如果我们不做性能测试那么就会很容易忽略这个问题。当以developer box中一个node.js实例来作为标准测试,这个同步调用将导致性能从每秒上千次的请求降至只有几十个。


2.关闭套接字池


Node.js的会自动地使用套接字池:默认地,。虽然套接字的重复使用可能会让资源的增加在控制之下,但如果你需要处理许多数据来自于同一主机的并发请求时,将会导致一系列的瓶颈。在这种情况下,增大maxSockets 的值或关闭套接字池是个好主意:

// Disable socket pooling

 

var http = require('http');

var options = {.....};

options.agent = false;

var req = http.request(options)


3.不要让静态资源使用Node.js


对于css和图片等静态资源,用标准的WebServer而不是Node.js。例如,领英移动使用的是nginx。我们同时还利用内容传递网络(CDNs),它能将世界范围内的静态资拷贝到服务器上。这有两个好处:(1)能减少我们node.js服务器的负载量(2)CDNs可以让静态内容在离用户较近的服务器上传递,以此来减少等待时间。


4.在客户端渲染


让我们快速比较一下服务器渲染和客户端渲染的区别。如果我们用node.js在服务器端渲染,对于每个请求我们都会回送像下面这样的HTML页面:

<!-- An example of a simple webpage rendered entirely server side -->

 

<!DOCTYPE html>

<html>

<head>

<title>LinkedIn Mobile</title>

</head>

<body>

<div>

<img src="http://mobile-cdn.linkedin.com/images/linkedin.png" alt="LinkedIn"/>

</div>

<div>

Hello John!

</div>

</body>

</html>

请注意观察这个页面所有的内容,除了用户的名字,其余都是静态内容:对于每个用户和页面重载内容都是一样的。因此更有效的作法是让Node.js仅以JSON形式返回页面需要的动态内容。

{"name": "John"}

页面的其余部分—所有静态的HTML标记-能放在JavaScript模板中(比如):

<!-- An example of a JavaScript template that can be rendered client side -->

 

<!DOCTYPE html>

<html>

<head>

<title>LinkedIn Mobile</title>

</head>

<body>

<div>

<img src="http://mobile-cdn.linkedin.com/images/linkedin.png" alt="LinkedIn"/>

</div>

<div>

Hello <%= name %>!

</div>

</body>

</html>

性能的提升来自于这些地方:如第三点所说,静态JavaScript模板能通过webserver(比如nginx)在服务器端提供,或者用更好的CDN来实现。此外,JavaScript模板能缓存在浏览器中或存储在本地,所有初始页面加载以后,唯一需要发送给客户端的数据就是JSON,这将是最有效果的。这个方法能极大性地减少CPU,IO,和Node.js的负载量。


5.使用gzip


许多服务器和客户端支持gzip来压缩请求和应答。无论是应答客户端还是向远程服务器发送请求,请确保充分使用它。

极快的node.js:来自领英(LinkedIn)移动的10个性能技巧

6.并行化


试着让你所有的阻塞操作-向远程服务发送请求,DB调用,文件系统访问并行化。这将能减少最慢的阻塞操作的等待时间,而不是所有阻塞操作的等待时间。为了保持回调和错误处理的干净,我们使用Step来控制流量。

极快的node.js:来自领英(LinkedIn)移动的10个性能技巧

7.Session自由化


领英移动使用Express框架来管理请求/应答周期。许多express的例子都包含如下的配置:

app.use(express.session({ secret: "keyboard cat" }));
Copyright © 2012-2017 领英助手(Linkedin助手 | Linkedin Assistant) 版权所有
  • QQ交谈
  • 售前咨询
  • 客服热线:
  • 1752755579
  • 扫一扫 体验微信营销