欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解node中創(chuàng)建服務(wù)進程

 更新時間:2017年05月09日 08:27:06   作者:royalrover  
本篇文章主要介紹了詳解node中創(chuàng)建服務(wù)進程,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

背景

在node工程部署中,常常涉及到三方:本地客戶端、跳板機和服務(wù)器(集群)。在通過git觸發(fā)gitlab hook腳本后,需要在跳板機中執(zhí)行相應的ssh命令執(zhí)行shell文件啟動node服務(wù)器,這需要使用一個常用的命令setsid,這樣當ssh命令執(zhí)行完畢shell退出后,node服務(wù)器仍正常運行,此時node服務(wù)進程就是一個最典型的daemon進程(后臺服務(wù)進程)。

那么,在node項目中,如何創(chuàng)建一個daemon進程呢?最簡單的方式,其實就是采用類似上文中介紹的方式:

復制代碼 代碼如下:

require('child_process').exec('setsid node app.js >/dev/null 2>&1 &');

這樣可以通過執(zhí)行shell的方式實現(xiàn)daemon進程。不過本文的重點并不是介紹這種“命令行”的方式實現(xiàn)daemon進程,而且本文會詳細講述daemon進程的創(chuàng)建原理,且看下文。

目標

在當前業(yè)務(wù)中,之所以需要創(chuàng)建daemon進程就是為了保證中斷創(chuàng)建該進程的父進程(ctrl+c)或者父進程執(zhí)行完畢后并不影響daemon進程的執(zhí)行。下文介紹兩種實現(xiàn)方式,實現(xiàn)原理細節(jié)上有些出入。

下文中的所有討論都是在linux環(huán)境下進行。

實現(xiàn)一

在linux系統(tǒng)中,父進程創(chuàng)建出子進程,此時父進程若退出,此時子進程則變?yōu)楣聝哼M程,其ppid變?yōu)?,即成為init進程的子進程。在node環(huán)境下,如果不針對子進程的stdio做一些特殊處理父進程其實不會真正退出,而是直到子進程執(zhí)行完畢后再退出。之所以出現(xiàn)這種情況是由于node創(chuàng)建子進程時默認會通過pipe方式將子進程的輸出導流到父進程的stream中(childProcess.stdout、childProcess.stderr),提供在父進程中輸出子進程消息的能力。

因此,解決此種問題可給子進程的stdio重新賦值:

file: parent.js

let cp = require('child_process');
const sp = cp.spawn('node',['./c.js'],{
  stdio: [process.stdin,process.stdout,process.stderr]
});

setTimeout(()=>{console.log('parent out')},5000);

--------------
file: c.js

setTimeout(()=>{
  console.log('children exit');
},10000)

通過在parent.js中設(shè)置子進程的stdio為當前終端(其實繼承了父進程的stdio),這樣父進程在5s后退出,此時子進程的ppid變?yōu)?,10s后子進程退出。

上述實現(xiàn)只滿足“父進程正常退出,子進程成為守護進程”的情況,一旦通過“ctrl+c”的方式終端父進程,子進程仍會退出,這還是與node底層實現(xiàn)有關(guān)。默認“ctrl+c”觸發(fā)SIGINT信號,父進程接受信號后發(fā)送給子進程,如果子進程存在SIGINT偵聽函數(shù),則會執(zhí)行該函數(shù),否則執(zhí)行exit系統(tǒng)調(diào)用子進程退出。因此,如果要讓子進程在接收到SIGINT信號不退出,只需要不作處理即可:

file: c.js

process.on('SIGINT',function(){
  console.log('child sigint');
});

setTimeout(()=>{
  console.log('children exit');
},10000)

以上實現(xiàn),可以滿足我們最初指定的目標:“父進程退出或者中斷,子進程仍正常運行”。

 實現(xiàn)二

node官方提供了創(chuàng)建daemon進程的相關(guān)API,如果不仔細閱讀文檔還真不容易發(fā)現(xiàn)該特性。在child_process模塊中有個spawn函數(shù),通過spawn可以執(zhí)行shell命令及其相關(guān)選項,同時spawn提供了創(chuàng)建子進程的一些選項,其中“detached”選項則與我們的需求密切相關(guān)。

detached選項可以讓node原生幫我們創(chuàng)建一個daemon進程,設(shè)置datached為true可以創(chuàng)建一個新的session和進程組,子進程的pid為新創(chuàng)建進程組的組pid,這與setsid起到相同的作用。此時的子進程已經(jīng)和其父進程屬于兩個session,因此父進程的退出和中斷信號不會傳遞給子進程,子進程不會接受到父進程的中斷信號自然也不會退出。當父進程結(jié)束之后,子進程變?yōu)楣聝哼M程從而被init進程接收,ppid設(shè)置為1。

file: parent.js

let cp = require('child_process');
const sp = cp.spawn('node',['./c.js'],{
  detached: true,
  stdio: [process.stdin,process.stdout,process.stdout]
});

sp.unref();
setTimeout(()=>{console.log('parent out')},5000);

----------------------
file: c.js

setTimeout(()=>{
  console.log('children exit');
},100000)

此時,c.js文件并未設(shè)置SIGINT事件偵聽函數(shù),在父進程中斷后仍會正常運行,正是由于其和父進程分屬于兩個session。

在parent.js文件中設(shè)置了sp.unref()函數(shù),目的是“避免父進程等待子進程退出”。那么為何會出現(xiàn)上述情況呢?這與node的事件循環(huán)有關(guān),讓父進程的事件循環(huán)排除對ChildProcess子進程對象的引用,可以使父進程單獨退出。

總結(jié)

為什么上文介紹的兩個方法都可以實現(xiàn)daemon進程呢?這還得回到系統(tǒng)層面進行分析。在linux系統(tǒng)創(chuàng)建一個daemon進程需要幾個步驟:

1.父進程創(chuàng)建子進程,父進程退出,讓子進程成為孤兒進程,ppid=1

2.通過setsid命令或函數(shù)在子進程中創(chuàng)建新的會話和進程組

3.設(shè)置當前目錄

4.設(shè)置文件權(quán)限,并關(guān)閉父進程繼承打開的fd

所謂會話和進程組,則是在linux多任務(wù)多用戶下的概念。不同會話的進程無法通過通信,因此父子進程相隔離。而執(zhí)行setsid命令則讓子進程有了新的特性:

  1. 子進程脫離父進程所在的session控制,兩者獨立存在互不影響
  2. 子進程脫離父進程所在的進程組
  3. 子進程脫離原先的命令行終端,終端退出不影響子進程

下面再回顧方法一與方法二的區(qū)別,發(fā)現(xiàn)方法一其實并不是真正的daemon進程,只是通過偵聽相關(guān)中斷信號并設(shè)置nop函數(shù)(不執(zhí)行默認的中斷行為)保證子進程繼續(xù)運行而已;而方法二則是標準的deamon進程創(chuàng)建方式,優(yōu)先使用!

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • 詳解如何使用Node.js實現(xiàn)熱重載頁面

    詳解如何使用Node.js實現(xiàn)熱重載頁面

    這篇文章主要介紹了詳解如何使用Node.js實現(xiàn)熱重載頁面,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-05-05
  • node.js同步/異步文件讀寫-fs,Stream文件流操作實例詳解

    node.js同步/異步文件讀寫-fs,Stream文件流操作實例詳解

    這篇文章主要介紹了node.js同步/異步文件讀寫-fs,Stream文件流操作,結(jié)合實例形式詳細分析了node.js針對文件的同步/異步讀寫與文件流相關(guān)操作技巧,需要的朋友可以參考下
    2023-06-06
  • nodejs6下使用koa2框架實例

    nodejs6下使用koa2框架實例

    本篇文章主要介紹了nodejs6下使用koa2框架實例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-05-05
  • 淺談Node.js 子進程與應用場景

    淺談Node.js 子進程與應用場景

    這篇文章主要介紹了淺談Node.js 子進程與應用場景,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-01-01
  • jwt在node中的應用實踐(安裝配置封裝)

    jwt在node中的應用實踐(安裝配置封裝)

    這篇文章主要為大家介紹了jwt在node中的應用實踐包括安裝配置封裝,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-09-09
  • 基于 Node.js 實現(xiàn)前后端分離

    基于 Node.js 實現(xiàn)前后端分離

    為了解決傳統(tǒng)Web開發(fā)模式帶來的各種問題,我們進行了許多嘗試,但由于前/后端的物理鴻溝,嘗試的方案都大同小異。痛定思痛,今天我們重新思考了“前后端”的定義,引入前端同學都熟悉的NodeJS,試圖探索一條全新的前后端分離模式。
    2016-04-04
  • 進階之初探nodeJS

    進階之初探nodeJS

    本文主要介紹了nodeJS的相關(guān)知識。具有很好的參考價值,下面跟著小編一起來看下吧
    2017-01-01
  • 如何構(gòu)建一個?NodeJS?影院微服務(wù)并使用?Docker?部署

    如何構(gòu)建一個?NodeJS?影院微服務(wù)并使用?Docker?部署

    微服務(wù)是一個單獨的自包含單元,與其他許多單元一起構(gòu)成一個大型應用程序,這篇文章主要介紹了如何構(gòu)建一個NodeJS影院微服務(wù)并使用Docker部署,在這個系列中,將構(gòu)建一個 NodeJS 微服務(wù),并使用 Docker Swarm 集群進行部署,需要的朋友可以參考下
    2023-08-08
  • node.js中的fs.closeSync方法使用說明

    node.js中的fs.closeSync方法使用說明

    這篇文章主要介紹了node.js中的fs.closeSync方法使用說明,本文介紹了fs.closeSync方法說明、語法、接收參數(shù)、使用實例和實現(xiàn)源碼,需要的朋友可以參考下
    2014-12-12
  • 基于nodejs res.end和res.send的區(qū)別

    基于nodejs res.end和res.send的區(qū)別

    今天小編就為大家分享一篇基于nodejs res.end和res.send的區(qū)別,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-05-05

最新評論