在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

介紹Node.js應用全鏈路信息獲取的方法

OSC開源社區 ? 來源:vivo互聯網技術 ? 2023-02-10 11:21 ? 次閱讀

全鏈路追蹤技術的兩個核心要素分別是 全鏈路信息獲取全鏈路信息存儲展示

一、Node.js 應用全鏈路追蹤系統

目前行業內, 不考慮 Serverless 的情況下,主流的 Node.js 架構設計主要有以下兩種方案:

通用架構:只做 ssr 和 bff,不做服務器和微服務;

全場景架構:包含 ssr、bff、服務器、微服務。

上述兩種方案對應的架構說明圖如下圖所示:

f4fc0c64-a8d8-11ed-bfe3-dac502259ad0.png

在上述兩種通用架構中,nodejs 都會面臨一個問題,那就是:

在請求鏈路越來越長,調用服務越來越多,其中還包含各種微服務調用的情況下,出現了以下訴求:

如何在請求發生異常時快速定義問題所在;

如何在請求響應慢的時候快速找出慢的原因;

如何通過日志文件快速定位問題的根本原因。

我們要解決上述訴求,就需要有一種技術,將每個請求的關鍵信息聚合起來,并且將所有請求鏈路串聯起來。讓我們可以知道一個請求中包含了幾次服務、微服務請求的調用,某次服務、微服務調用在哪個請求的上下文。

這種技術,就是Node.js應用全鏈路追蹤。它是 Node.js 在涉及到復雜服務端業務場景中,必不可少的技術保障。

綜上,我們需要Node.js應用全鏈路追蹤,說完為什么需要后,下面將介紹如何做Node.js應用的全鏈路信息獲取。

二、全鏈路信息獲取

全鏈路信息獲取,是全鏈路追蹤技術中最重要的一環。只有打通了全鏈路信息獲取,才會有后續的存儲展示流程。

對于多線程語言如 JavaPython 來說,做全鏈路信息獲取有線程上下文如 ThreadLocal 這種利器相助。而對于Node.js來說,由于單線程和基于IO回調的方式來完成異步操作,所以在全鏈路信息獲取上存在天然獲取難度大的問題。那么如何解決這個問題呢?

三、業界方案

由于 Node.js 單線程,非阻塞 IO 的設計思想。在全鏈路信息獲取上,到目前為止,主要有以下 4 種方案:

domain: node api

zone.js: Angular 社區產物;

顯式傳遞:手動傳遞、中間件掛載;

Async Hooks:node api;

而上述 4 個方案中, domain 由于存在嚴重的內存泄漏,已經被廢棄了;zone.js 實現方式非常暴力、API比較晦澀、最關鍵的缺點是 monkey patch 只能 mock api ,不能 mock language;顯式傳遞又過于繁瑣和具有侵入性;綜合比較下來,效果最好的方案就是第四種方案,這種方案有如下優點:

node 8.x 新加的一個核心模塊,Node 官方維護者也在使用,不存在內存泄漏;

非常適合實現隱式的鏈路跟蹤,入侵小,目前隱式跟蹤的最優解;

提供了 API 來追蹤 node 中異步資源的生命周期;

借助 async_hook 實現上下文的關聯關系;

優點說完了,下面我們就來介紹如何通過 Async Hooks 來獲取全鏈路信息。

四、Async Hooks【異步鉤子】

4.1 Async Hooks 概念

Async Hooks 是 Node.js v8.x 版本新增加的一個核心模塊,它提供了 API 用來追蹤 Node.js 中異步資源的生命周期,可幫助我們正確追蹤異步調用的處理邏輯及關系。在代碼中,只需要寫 import asyncHook from 'async_hooks' 即可引入 async_hooks 模塊。

一句話概括:async_hooks 用來追蹤 Node.js 中異步資源的生命周期。

目前 Node.js 的穩定版本是 v14.17.0 。我們通過一張圖看下 Async Hooks 不同版本的 api 差異。如下圖所示:

f5352206-a8d8-11ed-bfe3-dac502259ad0.png

從圖中可以看到該 api 變動較大。這是因為從 8 版本到 14 版本,async_hooks 依舊還是 Stability: 1 - Experimental

Stability: 1 - Experimental :該特性仍處于開發中,且未來改變時不做向后兼容,甚至可能被移除。不建議在生產環境中使用該特性。

但是沒關系,要相信官方團隊,這里我們的全鏈路信息獲取方案是基于 Node v9.x 版本 api 實現的。對于 Async Hooks api 介紹和基本使用, 大家可以閱讀官方文檔,下文會闡述對核心知識的理解。

下面我們將系統介紹基于 Async Hooks 的全鏈路信息獲取方案的設計和實現,下文統稱為 zone-context 。

4.2 理解 async_hooks 核心知識

在介紹 zone-context 之前,要對 async_hooks 的核心知識有正確的理解,這里做了一個總結,有如下6點:

每一個函數(不論異步還是同步)都會提供一個上下文, 我們稱之為 async scope ,這個認知對理解 async_hooks 非常重要;

每一個 async scope 中都有一個 asyncId ,它是當前 async scope 的標志,同一個的 async scope 中 asyncId 必然相同,每個異步資源在創建時, asyncId 自動遞增,全局唯一;

每一個 async scope 中都有一個 triggerAsyncId ,用來表示當前函數是由哪個 async scope 觸發生成的;

通過 asyncId 和 triggerAsyncId 我們可以追蹤整個異步的調用關系及鏈路,這個是全鏈路追蹤的核心;

通過 async_hooks.createHook 函數來注冊關于每個異步資源在生命周期中發生的 init 等相關事件的監聽函數;

同一個 async scope 可能會被調用及執行多次,不管執行多少次,其 asyncId 必然相同,通過監聽函數,我們很方便追蹤其執行的次數、時間以及上下文關系。

上述6點知識對于理解 async_hooks 是非常重要的。正是因為這些特性,才使得 async_hooks 能夠優秀的完成Node.js 應用全鏈路信息獲取。

到這里,下面就要介紹 zone-context 的設計和實現了,請和我一起往下看。

五、zone-context

5.1 架構設計

整體架構設計如下圖所示:

f595ecee-a8d8-11ed-bfe3-dac502259ad0.png

核心邏輯如下:異步資源(調用)創建后,會被 async_hooks 監聽到。監聽到后,對獲取到的異步資源信息進行處理加工,整合成需要的數據結構,整合后,將數據存儲到 invoke tree 中。在異步資源結束時,觸發 gc 操作,對 invoke tree 中不再有用的數據進行刪除回收。

從上述核心邏輯中,我們可以知道,此架構設計需要實現以下三個功能:

異步資源(調用)監聽

invoke tree

gc

下面開始逐個介紹上述三個功能的實現。

5.2 異步資源(調用)監聽

如何做到監聽異步調用呢?

這里用到了 async_hooks (追蹤 Node.js 異步資源的生命周期)代碼實現如下:

asyncHook
  .createHook({
    init(asyncId, type, triggerAsyncId) {
      // 異步資源創建(調用)時觸發該事件
    },
  })
  .enable()

是不是發現此功能實現非常簡單,是的哦,就可以對所有異步操作進行追蹤了。

在理解 async_hooks 核心知識中,我們提到了通過 asyncId 和 triggerAsyncId 可以追蹤整個異步的調用關系及鏈路。現在大家看 init 中的參數,會發現, asyncId 和triggerAsyncId 都存在,而且是隱式傳遞,不需要手動傳入。這樣,我們在每次異步調用時,都能在 init 事件中,拿到這兩個值。invoke tree 功能的實現,離不開這兩個參數。

介紹完異步調用監聽,下面將介紹 invoke tree 的實現。

5.3 invoke tree 設計和異步調用監聽結合

5.3.1 設計

invoke tree 整體設計思路如下圖所示:

f5e821bc-a8d8-11ed-bfe3-dac502259ad0.png

具體代碼如下:

interface ITree {
  [key: string]: {
    // 調用鏈路上第一個異步資源asyncId
    rootId: number
    // 異步資源的triggerAsyncId
    pid: number
    // 異步資源中所包含的異步資源asyncId
    children: Array
  }
}
 
const invokeTree: ITree = {}

創建一個大的對象 invokeTree, 每一個屬性代表一個異步資源的完整調用鏈路。屬性的key和value代表含義如下:

屬性的 key 是代表這個異步資源的 asyncId。

屬性的 value 是代表這個異步資源經過的所有鏈路信息聚合對象,該對象中的各屬性含義請看上面代碼中的注釋進行理解。

通過這種設計,就能拿到任何一個異步資源在整個請求鏈路中的關鍵信息。收集根節點上下文。

5.3.2和異步調用監聽結合

雖然 invoke tree 設計好了。但是如何在 異步調用監聽的 init 事件中,將 asyncId 、 triggerAsyncId 和 invokeTree 關聯起來呢?

代碼如下:

asyncHook
  .createHook({
    init(asyncId, type, triggerAsyncId) {
      // 尋找父節點
      const parent = invokeTree[triggerAsyncId]
      if (parent) {
        invokeTree[asyncId] = {
          pid: triggerAsyncId,
          rootId: parent.rootId,
          children: [],
        }
        // 將當前節點asyncId值保存到父節點的children數組中
        invokeTree[triggerAsyncId].children.push(asyncId)
      }
    }
  })
  .enable()

大家看上面代碼,整個代碼大致有以下幾個步驟:

當監聽到異步調用的時候,會先去 invokeTree 對象中查找是否含有 key 為 triggerAsyncId 的屬性;

有的話,說明該異步調用在該追蹤鏈路中,則進行存儲操作,將 asyncId 當成 key , 屬性值是一個對象,包含三個屬性,分別是 pid、rootId、children , 具體含義上文已說過;

沒有的話,說明該異步調用不在該追蹤鏈路中。則不進行任何操作,如把數據存入 invokeTree 對象;

將當前異步調用 asyncId 存入到 invokeTree 中 key 為 triggerAsyncId 的 children 屬性中。

至此,invoke tree 的設計、和異步調用監聽如何結合,已經介紹完了。下面將介紹 gc 功能的設計和實現。

5.4gc

5.4.1 目的

我們知道,異步調用次數是非常多的,如果不做 gc 操作,那么 invoke tree 會越來越大,node應用的內存會被這些數據慢慢占滿,所以需要對 invoke tree 進行垃圾回收。

5.4.2設計

gc 的設計思想主要如下:當異步資源結束的時候,觸發垃圾回收,尋找此異步資源觸發的所有異步資源,然后按照此邏輯遞歸查找,直到找出所有可回收的異步資源。

話不多說,直接上代碼, gc 代碼如下:

interface IRoot {
  [key: string]: Object
}
 
// 收集根節點上下文
const root: IRoot = {}
 
function gc(rootId: number) {
  if (!root[rootId]) {
    return
  }
 
  // 遞歸收集所有節點id
  const collectionAllNodeId = (rootId: number) => {
    const {children} = invokeTree[rootId]
    let allNodeId = [...children]
    for (let id of children) {
      // 去重
      allNodeId = [...allNodeId, ...collectionAllNodeId(id)]
    }
    return allNodeId
  }
 
  const allNodes = collectionAllNodeId(rootId)
 
  for (let id of allNodes) {
    delete invokeTree[id]
  }
 
  delete invokeTree[rootId]
  delete root[rootId]
}

gc 核心邏輯:用 collectionAllNodeId 遞歸查找所有可回收的異步資源( id )。然后再刪除 invokeTree 中以這些 id 為 key 的屬性。最后刪除根節點。

大家看到了聲明對象 root ,這個是什么呢?

root 其實是我們對某個異步調用進行監聽時,設置的一個根節點對象,這個節點對象可以手動傳入一些鏈路信息,這樣可以為全鏈路追蹤增加其他追蹤信息,如錯誤信息、耗時時間等。

5.5萬事具備,只欠東風

我們的異步事件監聽設計好了, invoke tree 設計好了,gc 也設計好了。那么如何將他們串聯起來呢?比如我們要監聽某一個異步資源,那么我們要怎樣才能把 invoke tree 和異步資源結合起來呢?

這里需要三個函數來完成結合,分別是 ZoneContext setZoneContextgetZoneContext。下面來一一介紹下這三個函數:

5.5.1 ZoneContext

這是一個工廠函數,用來創建異步資源實例的,代碼如下所示:

// 工廠函數
async function ZoneContext(fn: Function) {
  // 初始化異步資源實例
  const asyncResource = new asyncHook.AsyncResource('ZoneContext')
  let rootId = -1
  return asyncResource.runInAsyncScope(async () => {
    try {
      rootId = asyncHook.executionAsyncId()
      // 保存 rootId 上下文
      root[rootId] = {}
      // 初始化 invokeTree
      invokeTree[rootId] = {
        pid: -1, // rootId 的 triggerAsyncId 默認是 -1
        rootId,
        children: [],
      }
      // 執行異步調用
      await fn()
    } finally {
      gc(rootId)
    }
  })
}

大家會發現,在此函數中,有這樣一行代碼:

const asyncResource = new asyncHook.AsyncResource('ZoneContext')

這行代碼是什么含義呢?

它是指我們創建了一個名為 ZoneContext 的異步資源實例,可以通過該實例的屬性方法來更加精細的控制異步資源。

執行 asyncResource.runInAsyncScope 方法有什么用處呢?

調用該實例的 runInAsyncScope方法,在runInAsyncScope 方法中包裹要傳入的異步調用。可以保證在這個資源( fn )的異步作用域下,所執行的代碼都是可追蹤到我們設置的 invokeTree 中,達到更加精細控制異步調用的目的。在執行完后,進行gc調用,完成內存回收。

5.5.2setZoneContext

用來給異步調用設置額外的跟蹤信息。代碼如下:

function setZoneContext(obj: Object) {
  const curId = asyncHook.executionAsyncId()
  let root = findRootVal(curId)
  Object.assign(root, obj)
}

通過 Object.assign(root, obj) 將傳入的 obj 賦值給 root 對象中, key 為 curId 的屬性。這樣就可以給我們想跟蹤的異步調用設置想要跟蹤的信息。

5.5.3 getZoneContext

用來拿到異步調的 rootId 的屬性值。代碼如下:

function findRootVal(asyncId: number) {
  const node = invokeTree[asyncId]
  return node ? root[node.rootId] : null
}
function getZoneContext() {
  const curId = asyncHook.executionAsyncId()
  return findRootVal(curId)
}

通過給 findRootVal 函數傳入 asyncId 來拿到 root 對象中 key 為 rootId 的屬性值。這樣就可以拿到當初我們設置的想要跟蹤的信息了,完成一個閉環。

至此,我們將 Node.js應用全鏈路信息獲取的核心設計和實現闡述完了。邏輯上有點抽象,需要多去思考和理解,才能對全鏈路追蹤信息獲取有一個更加深刻的掌握。

最后,我們使用本次全鏈路追蹤的設計實現來展示一個追蹤 demo 。

5.6 使用 zone-context

5.6.1 確定異步調用嵌套關系

為了更好的闡述異步調用嵌套關系,這里進行了簡化,沒有輸出 invoke tree 。例子代碼如下:

// 對異步調用A函數進行追蹤
ZoneContext(async () => {
  await A()
})
 
// 異步調用A函數中執行異步調用B函數
async function A() {
  // 輸出 A 函數的 asyncId
  fs.writeSync(1, `A 函數的 asyncId -> ${asyncHook.executionAsyncId()}
`)
  Promise.resolve().then(() => {
    // 輸出 A 函數中執行異步調用時的 asyncId
    fs.writeSync(1, `A 執行異步 promiseC 時 asyncId 為 -> ${asyncHook.executionAsyncId()}
`)
    B()
  })
}
 
// 異步調用B函數中執行異步調用C函數
async function B() {
  // 輸出 B 函數的 asyncId
  fs.writeSync(1, `B 函數的 asyncId -> ${asyncHook.executionAsyncId()}
`)
  Promise.resolve().then(() => {
    // 輸出 B 函數中執行異步調用時的 asyncId
    fs.writeSync(1, `B 執行異步 promiseC 時 asyncId 為 -> ${asyncHook.executionAsyncId()}
`)
    C()
  })
}
 
// 異步調用C函數
function C() {
  const obj = getZoneContext()
  // 輸出 C 函數的 asyncId
  fs.writeSync(1, `C 函數的 asyncId -> ${asyncHook.executionAsyncId()}
`)
  Promise.resolve().then(() => {
    // 輸出 C 函數中執行異步調用時的 asyncId
    fs.writeSync(1, `C 執行異步 promiseC 時 asyncId 為 -> ${asyncHook.executionAsyncId()}
`)
  })
}

輸出結果為:

A 函數的 asyncId -> 3
A 執行異步 promiseA 時 asyncId 為 -> 8
B 函數的 asyncId -> 8
B 執行異步 promiseB 時 asyncId 為 -> 13
C 函數的 asyncId -> 13
C 執行異步 promiseC 時 asyncId 為 -> 16

只看輸出結果就可以推出以下信息:

A 函數執行異步調用后, asyncId 為 8 ,而 B 函數的 asyncId 是 8 ,這說明, B 函數是被 A 函數 調用;

B 函數執行異步調用后, asyncId 為 13 ,而 C 函數的 asyncId 是 13 ,這說明, C 函數是被 B 函數 調用;

C 函數執行異步調用后, asyncId 為 16 , 不再有其他函數的 asyncId 是 16 ,這說明, C 函數中沒有調用其他函數;

綜合上面三點,可以知道,此鏈路的異步調用嵌套關系為:A —> B -> C;

至此,我們可以清晰快速的知道誰被誰調用,誰又調用了誰。

5.6.2 額外設置追蹤信息

在上面例子代碼的基礎下,增加以下代碼:

ZoneContext(async () => {
  const ctx = { msg: '全鏈路追蹤信息', code: 1 }
  setZoneContext(ctx)
  await A()
})
 
function A() {
  // 代碼同上個demo
}
 
function B() {
  // 代碼同上個demo
  D()
}
 
// 異步調用C函數
function C() {
  const obj = getZoneContext()
  Promise.resolve().then(() => {
    fs.writeSync(1, `getZoneContext in C -> ${JSON.stringify(obj)}
`)
  })
}
 
// 同步調用函數D
function D() {
  const obj = getZoneContext()
  fs.writeSync(1, `getZoneContext in D -> ${JSON.stringify(obj)}
`)
}

輸出以下內容:

呈現代碼宏出錯:參數

'com.atlassian.confluence.ext.code.render.InvalidValueException'的值無效。

getZoneContext in D -> {"msg":"全鏈路追蹤信息","code":1}

getZoneContext in C-> {"msg":"全鏈路追蹤信息","code":1}

可以發現, 執行 A 函數前設置的追蹤信息后,調用 A 函數, A 函數中調用 B 函數, B 函數中調用 C 函數和 D 函數。在 C 函數和 D 函數中,都能訪問到設置的追蹤信息。

這說明,在定位分析嵌套的異步調用問題時,通過 getZoneContext 拿到頂層設置的關鍵追蹤信息。可以很快回溯出,某個嵌套異步調用出現的異常,

是由頂層的某個異步調用異常所導致的。

5.6.3 追蹤信息大而全的 invoke tree

例子代碼如下:

ZoneContext(async () => {
  await A()
})
async function A() {
  Promise.resolve().then(() => {
    fs.writeSync(1, `A 函數執行異步調用時的 invokeTree -> ${JSON.stringify(invokeTree)}
`)
    B()
  })
}
async function B() {
  Promise.resolve().then(() => {
    fs.writeSync(1, `B 函數執行時的 invokeTree -> ${JSON.stringify(invokeTree)}
`)
  })
}

輸出結果如下:

A 函數執行異步調用時的 invokeTree -> {"3":{"pid":-1,"rootId":3,"children":[5,6,7]},"5":{"pid":3,"rootId":3,"children":[10]},"6":{"pid":3,"rootId":3,"children":[9]},"7":{"pid":3,"rootId":3,"children":[8]},"8":{"pid":7,"rootId":3,"children":[]},"9":{"pid":6,"rootId":3,"children":[]},"10":{"pid":5,"rootId":3,"children":[]}}
 
B 函數執行異步調用時的 invokeTree -> {"3":{"pid":-1,"rootId":3,"children":[5,6,7]},"5":{"pid":3,"rootId":3,"children":[10]},"6":{"pid":3,"rootId":3,"children":[9]},"7":{"pid":3,"rootId":3,"children":[8]},"8":{"pid":7,"rootId":3,"children":[11,12]},"9":{"pid":6,"rootId":3,"children":[]},"10":{"pid":5,"rootId":3,"children":[]},"11":{"pid":8,"rootId":3,"children":[]},"12":{"pid":8,"rootId":3,"children":[13]},"13":{"pid":12,"rootId":3,"children":[]}}

根據輸出結果可以推出以下信息:

1、此異步調用鏈路的 rootId (初始 asyncId ,也是頂層節點值) 是 3

2、函數執行異步調用時,其調用鏈路如下圖所示:

f5f9d916-a8d8-11ed-bfe3-dac502259ad0.png

3、函數執行異步調用時,其調用鏈路如下圖所示:

f61ee440-a8d8-11ed-bfe3-dac502259ad0.png

從調用鏈路圖就可以清晰看出所有異步調用之間的相互關系和順序。為異步調用的各種問題排查和性能分析提供了強有力的技術支持。

六、總結

到這,關于Node.js 應用全鏈路信息獲取的設計、實現和案例演示就介紹完了。全鏈路信息獲取是全鏈路追蹤系統中最重要的一環,當信息獲取搞定后,下一步就是全鏈路信息存儲展示。






審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • SSR
    SSR
    +關注

    關注

    0

    文章

    83

    瀏覽量

    17781
  • JAVA語言
    +關注

    關注

    0

    文章

    138

    瀏覽量

    20112

原文標題:Node.js應用全鏈路追蹤技術——[全鏈路信息獲取]

文章出處:【微信號:OSC開源社區,微信公眾號:OSC開源社區】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    Node.js 給前端帶來了什么

    端分工方式。讓Node.js來改變這一切  Node.js一發布,立刻在前端工程師中引起了軒然大波,前端工程師們幾乎立刻對這一項技術表露出了相當大的熱情和期待。上一次一種技術能被整個前端界如此關注那還
    發表于 05-06 14:23

    【Intel Edison試用體驗】XDK篇:Node.js操作SQLite3

    首先得用opkg安裝sqlite3,接著需使用npm安裝sqlite3的接口文件,以保證node.js與sqlite3可以連接,運行如下命令便可安裝: 接著就可以使用sqlite3了,新建一個
    發表于 07-25 09:39

    【orangepi zero試用體驗】安裝Node.JS運行環境與示例

    本帖最后由 eyecf 于 2016-12-24 17:51 編輯 本期要和大家介紹一下OrangePI zero安裝Node.js運行環境的方法,并運行一下vuejs的實例Node.j
    發表于 12-24 17:07

    基于Zetta(Node.js)的數據接收端server中,可接收數量不定傳感數據的IoT APP實現

    Zetta 是一個基于Node.js的IoT框架。上一篇 在基于Node.js的IoT框架Zetta中實現可變間隔發送數據 中對Zetta進行了簡要的介紹,以及給出了一個發送間隔可變數據的實現。這一
    發表于 08-09 17:57

    STM32MP157C-DK2有沒有辦法在不重建圖像的情況下安裝node.js

    我想在 STM32MP157C-DK2 上使用 node.js。我發現這篇文章https://community.st.com/s/question/0D50X0000B0xovG
    發表于 02-07 09:00

    深入淺出Node.js迷你書

    [InfoQ]深入淺出Node.js迷你書
    發表于 11-04 15:50 ?0次下載

    node.jsjs要點總結

    Node.js是一個面向服務器的框架,立足于Chrome強大的V8 JS引擎。盡管它由C++編寫而成,但是它及其應用是運行在JS上的。本文為開發者總結了4個Node.js要點。 1.
    發表于 10-13 10:39 ?0次下載

    三大方面對比Go語言和Node.js 誰更有優勢

    Node.js與Go語言一直是互聯網大戰中的主戰場,雖說按照普通的各項指標對比,那么這場戰爭可能在很長時間內都難分勝負,但我們還是決定嘗試對這二者做一些研究,并力求做出更準確的判斷。
    發表于 06-29 14:59 ?5175次閱讀

    node.js在訓練好的神經網絡模型識別圖像中物體的方法

    如何在Node.js環境下使用訓練好的神經網絡模型(Inception、SSD)識別圖像中的物體。
    的頭像 發表于 04-06 13:11 ?9173次閱讀

    第3部分:使用NoDE.JS的程序

    Implement MQTT to publish temperature data using Node.js*.
    的頭像 發表于 10-26 07:16 ?1788次閱讀

    Node.js 內存泄漏問題初探

    區:專門用來存儲引用類型的內存區域,比如對象、字符串和閉包在 Node.js 中,我們可以通過調用process.memoryUsage() 方法來來查詢內存使用情況。該函數返回值如下:memory
    的頭像 發表于 11-01 13:39 ?4740次閱讀

    Node.js的九大后端框架你都知道嗎

    Nest 是一個用于構建高效,可擴展的 Node.js 服務器端應用程序的框架。
    發表于 04-26 17:40 ?3280次閱讀
    <b class='flag-5'>Node.js</b>的九大后端框架你都知道嗎

    Node.js網頁控制的機器人小車

    電子發燒友網站提供《Node.js網頁控制的機器人小車.zip》資料免費下載
    發表于 02-08 16:06 ?0次下載
    <b class='flag-5'>Node.js</b>網頁控制的機器人小車

    node.js實戰源碼

    node.js實戰源碼
    發表于 05-16 18:06 ?1次下載

    Node.js小科普和Node.js安裝常見管理工具

    Node.js是一個JavaScript的運行環境,用來執行JavaScript代碼。 為什么會出現這么一個運行環境呢,從JavaScript研發初衷可以看出它是為了運行在瀏覽器中的,讓網頁交互更加
    的頭像 發表于 11-23 15:37 ?134次閱讀
    <b class='flag-5'>Node.js</b>小科普和<b class='flag-5'>Node.js</b>安裝常見管理工具
    主站蜘蛛池模板: 丁香网五月天| 精品你懂的| 夜夜春色| 一级做a爰片久久毛片毛片| 精品在线小视频| 国产特黄一级一片免费| 天天拍天天干天天操| 丁香六月婷婷在线| 久久夜色tv网站免费影院| 日本大片成人免费播放| 天天色图片| 日韩精品亚洲一级在线观看| 午夜黄色福利| 日本三级中文字幕| 色香蕉在线观看| 视色4se成人午夜精品| 欧美色频| 国产高清在线| 天天操天天擦| 人操人操| 日韩夜夜操| 日日添天天做天天爱| 高清不卡毛片免费观看| 国产精品1区2区3区| 额去鲁97在线观看视频| 久久精品网站免费观看| 精品二区| 中文字幕亚洲天堂| 四虎影院观看视频| 久久国产精品亚洲综合| 亚洲主播自拍| 爱爱视频天天干| 免费观看在线视频| h视频免费看| 亚洲合集综合久久性色| 亚洲欧美在线视频免费| 日本黄色三级视频| xvsr-365波多野结衣| 午夜影视体验区| 片免费观看在线看| 国产精品视频久久久久久|