精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

聊聊Nodejs的錯誤處理

開發 前端
當操作系統收到一個發給該socket的rst包的時候會執行tcp_reset,我們看到當socket處于發送syn包等待ack的時候,如果收到一個fin包,則會設置錯誤碼為ECONNREFUSED。我們輸出的正是這個錯誤碼。

[[392980]]

本文轉載自微信公眾號「編程雜技」,作者theanarkh。轉載本文請聯系編程雜技公眾號。  

本文以連接錯誤ECONNREFUSED為例,看看nodejs對錯誤處理的過程。

假設我們有以下代碼

  1. const net = require('net'); 
  2.  
  3. net.connect({port: 9999}) 

如果本機上沒有監聽9999端口,那么我們會得到以下輸出。

  1. events.js:170   
  2.        throw er; // Unhandled 'error' event   
  3.        ^   
  4.     
  5.  Error: connect ECONNREFUSED 127.0.0.1:9999   
  6.      at TCPConnectWrap.afterConnect [as oncomplete] (net.js:1088:14)   
  7.  Emitted 'error' event at:   
  8.      at emitErrorNT (internal/streams/destroy.js:91:8)   
  9.      at emitErrorAndCloseNT (internal/streams/destroy.js:59:3)   
  10.      at processTicksAndRejections (internal/process/task_queues.js:81:17)  

我們簡單看一下connect的調用流程。

  1. const req = new TCPConnectWrap();   
  2. req.oncomplete = afterConnect;   
  3. req.address = address;   
  4. req.port = port;   
  5. req.localAddress = localAddress;   
  6. req.localPort = localPort;   
  7. // 開始三次握手建立連接   
  8. err = self._handle.connect(req, address, port);  

接著我們看一下C++層connect的邏輯

  1. err = req_wrap->Dispatch(uv_tcp_connect,   
  2.                 &wrap->handle_,   
  3.                 reinterpret_cast(&addr),   
  4.                 AfterConnect);  

C++層直接調用Libuv的uv_tcp_connect,并且設置回調是AfterConnect。接著我們看libuv的實現。

  1. do {   
  2.     errno = 0;   
  3.     // 非阻塞調用   
  4.     r = connect(uv__stream_fd(handle), addr, addrlen);   
  5.   } while (r == -1 && errno == EINTR);   
  6.   // 連接錯誤,判斷錯誤碼   
  7.   if (r == -1 && errno != 0) {   
  8.     // 還在連接中,不是錯誤,等待連接完成,事件變成可讀   
  9.     if (errno == EINPROGRESS)   
  10.       ; /* not an error */   
  11.     else if (errno == ECONNREFUSED)   
  12.       // 連接被拒絕   
  13.       handle->delayed_error = UV__ERR(ECONNREFUSED);   
  14.     else   
  15.       return UV__ERR(errno);   
  16.   }   
  17.   uv__req_init(handle->loop, req, UV_CONNECT);   
  18.   req->cb = cb;   
  19.   req->handle = (uv_stream_t*) handle;   
  20.   QUEUE_INIT(&req->queue);   
  21.   // 掛載到handle,等待可寫事件   
  22.   handle->connect_req = req;   
  23. uv__io_start(handle->loop, &handle->io_watcher, POLLOUT);

我們看到Libuv以異步的方式調用操作系統,然后把request掛載到handle中,并且注冊等待可寫事件,當連接失敗的時候,就會執行uv__stream_io回調,我們看一下Libuv的處理(uv__stream_io)。

  1. getsockopt(uv__stream_fd(stream),   
  2.                SOL_SOCKET,   
  3.                SO_ERROR,   
  4.                &error,   
  5.                &errorsize);   
  6. error = UV__ERR(error);   
  7. if (req->cb)   
  8.     req->cb(req, error); 

獲取錯誤信息后回調C++層的AfterConnect。

  1. Localargv[5] = {   
  2.    Integer::New(env->isolate(), status),   
  3.    wrap->object(),   
  4.    req_wrap->object(),   
  5.    Boolean::New(env->isolate(), readable),   
  6.    Boolean::New(env->isolate(), writable)   
  7.  };   
  8.    
  9.  req_wrap->MakeCallback(env->oncomplete_string(), arraysize(argv), argv);   

接著調用JS層的oncomplete回調。

  1. const ex = exceptionWithHostPort(status,   
  2.                                  'connect',   
  3.                                  req.address,   
  4.                                  req.port,   
  5.                                  details);   
  6. if (details) {   
  7.   ex.localAddress = req.localAddress;   
  8.   ex.localPort = req.localPort;   
  9. }   
  10. // 銷毀socket   
  11. self.destroy(ex); 

exceptionWithHostPort構造錯誤信息,然后銷毀socket并且以ex為參數觸發error事件。我們看看uvExceptionWithHostPort的實現。

  1. function uvExceptionWithHostPort(err, syscall, address, port) {   
  2.   const [ code, uvmsg ] = uvErrmapGet(err) || uvUnmappedError;   
  3.   const message = `${syscall} ${code}: ${uvmsg}`;   
  4.   let details = '';   
  5.    
  6.   if (port && port > 0) {   
  7.     details = ` ${address}:${port}`;   
  8.   } else if (address) {   
  9.     details = ` ${address}`;   
  10.   }   
  11.   const tmpLimit = Error.stackTraceLimit;   
  12.   Error.stackTraceLimit = 0;   
  13.   const ex = new Error(`${message}${details}`);   
  14.   Error.stackTraceLimit = tmpLimit;   
  15.   ex.code = code;   
  16.   ex.errno = err;   
  17.   ex.syscall = syscall;   
  18.   ex.address = address;   
  19.   if (port) {   
  20.     ex.port = port;   
  21.   }   
  22.   // 獲取調用棧信息但不包括當前調用的函數uvExceptionWithHostPort,注入stack字段到ex中   
  23.   Error.captureStackTrace(ex, excludedStackFn || uvExceptionWithHostPort);   
  24.   return ex;   
  25. }   

我們看到錯誤信息主要通過uvErrmapGet獲取

  1. unction uvErrmapGet(name) {   
  2.    uvBinding = lazyUv();   
  3.    if (!uvBinding.errmap) {   
  4.      uvBinding.errmap = uvBinding.getErrorMap();   
  5.    }   
  6.    return uvBinding.errmap.get(name);   
  7.  }   
  8.     
  9.  function lazyUv() {   
  10.    if (!uvBinding) {   
  11.      uvBinding = internalBinding('uv');   
  12.    }   
  13.    return uvBinding;   
  14.  } 

繼續往下看,uvErrmapGet調用了C++層的uv模塊的getErrorMap。

  1. void GetErrMap(const FunctionCallbackInfo& args) {   
  2.   Environment* env = Environment::GetCurrent(args);   
  3.   Isolate* isolate = env->isolate();   
  4.   Localcontext = env->context();   
  5.    
  6.   Local 
  7.   // 從per_process::uv_errors_map中獲取錯誤信息   
  8.   size_t errors_len = arraysize(per_process::uv_errors_map);   
  9.   // 賦值   
  10.   for (size_t i = 0; i < errors_len; ++i) {   
  11.      // map的鍵是 uv_errors_map每個元素中的value,值是name和message 
  12.     const auto& error = per_process::uv_errors_map[i];   
  13.     Localarr[] = {OneByteString(isolate, error.name),   
  14.                           OneByteString(isolate, error.message)};  
  15.     if (err_map   
  16.             ->Set(context,   
  17.                   Integer::New(isolate, error.value),   
  18.                   Array::New(isolate, arr, arraysize(arr)))   
  19.             .IsEmpty()) {   
  20.       return;   
  21.     }   
  22.   }   
  23.    
  24.   args.GetReturnValue().Set(err_map);   

我們看到錯誤信息存在per_process::uv_errors_map中,我們看一下uv_errors_map的定義。

  1. struct UVError { 
  2.   int value; 
  3.   const charname
  4.   const char* message; 
  5. }; 
  6.  
  7. static const struct UVError uv_errors_map[] = {   
  8. #define V(name, message) {UV_##name, #name, message},   
  9.     UV_ERRNO_MAP(V)   
  10. #undef V   
  11. };   

UV_ERRNO_MAP宏展開后如下

  1. {UV_E2BIG, "E2BIG""argument list too long"},   
  2. {UV_EACCES, "EACCES""permission denied"},   
  3. {UV_EADDRINUSE, "EADDRINUSE""address already in use"},   
  4. ……  

所以導出到JS層的結果如下

  1. {   
  2.   // 鍵是一個數字,由Libuv定義,其實是封裝了操作系統的定義 
  3.   UV_ECONNREFUSED: ["ECONNREFUSED""connection refused"],     
  4.   UV_ECONNRESET: ["ECONNRESET""connection reset by peer"]    
  5.   ...    
  6. }  

Node.js最后會組裝這些信息返回給調用方。這就是我們輸出的錯誤信息。那么為什么會是ECONNREFUSED呢?我們看一下操作系統對于該錯誤碼的邏輯。

  1. static void tcp_reset(struct sock *sk)   
  2. {   
  3.     switch (sk->sk_state) {   
  4.         case TCP_SYN_SENT:   
  5.             sk->sk_err = ECONNREFUSED;   
  6.             break;   
  7.          // ... 
  8.     }   
  9.    
  10. }  

當操作系統收到一個發給該socket的rst包的時候會執行tcp_reset,我們看到當socket處于發送syn包等待ack的時候,如果收到一個fin包,則會設置錯誤碼為ECONNREFUSED。我們輸出的正是這個錯誤碼。

責任編輯:武曉燕 來源: 編程雜技
相關推薦

2021-04-29 09:02:44

語言Go 處理

2024-03-27 08:18:02

Spring映射HTML

2014-11-17 10:05:12

Go語言

2023-12-26 22:05:53

并發代碼goroutines

2022-11-16 08:41:43

2010-06-01 16:14:04

2009-08-05 16:04:50

2023-10-28 16:30:19

Golang開發

2009-06-19 16:20:14

ASP.NET錯誤處理

2023-10-08 20:31:18

React

2016-09-07 20:28:17

MySQL存儲數據庫

2016-08-19 10:41:42

Swift 2錯誤

2017-03-08 08:57:04

JavaScript錯誤堆棧

2011-05-25 10:26:42

ora-02069錯誤

2015-08-19 14:11:56

SQL Server錯誤處理

2017-04-06 14:40:29

JavaScript錯誤處理堆棧追蹤

2025-02-10 09:49:00

2021-09-13 07:53:31

Go錯誤處理

2025-03-31 00:29:44

2020-08-20 10:16:56

Golang錯誤處理數據
點贊
收藏

51CTO技術棧公眾號

国产一区二区三区四区三区四 | 欧美日韩中字一区| 久久久久久一级片| 污污网站在线免费观看| 狠狠久久综合婷婷不卡| 色播久久人人爽人人爽人人片视av| 国产在线资源一区| 探花视频在线观看| 久久精品国产www456c0m| 91麻豆精品国产91久久久 | 十八禁一区二区三区| 老牛嫩草一区二区三区日本| 久久夜色精品国产亚洲aⅴ| 白嫩情侣偷拍呻吟刺激| 国模私拍国内精品国内av| 亚洲成人第一页| 亚洲图片在线观看| 天天综合天天色| 九色porny丨国产精品| 97超级碰碰人国产在线观看| 亚洲欧美卡通动漫| 亚洲另类春色校园小说| 日韩一区二区影院| 国产视频一区二区视频| 电影k8一区二区三区久久| 中文字幕免费不卡| 国内精品一区二区| 99精品视频免费看| 蜜桃视频在线观看一区二区| 久久露脸国产精品| 亚洲 欧美 变态 另类 综合| 综合国产视频| 精品av综合导航| 小早川怜子一区二区三区| 高清电影一区| 午夜日韩在线观看| 国产成人生活片| 美女免费久久| 国产精品无码永久免费888| 久久久久一区二区三区| 亚洲精品久久久久久久久久| 国模娜娜一区二区三区| 国产欧美一区二区三区久久人妖| 天码人妻一区二区三区在线看| 欧美日韩亚洲三区| 久久成人一区二区| 欧美一级特黄高清视频| 成人羞羞视频在线看网址| 亚洲人成欧美中文字幕| 一区二区三区少妇| 同性恋视频一区| 亚洲成人精品久久| 午夜不卡久久精品无码免费| 日韩欧美高清一区二区三区| 91精品欧美一区二区三区综合在 | 国产精品2024| 91在线免费看片| 精品国产亚洲一区二区麻豆| 国产原创一区二区| 亚洲在线一区二区| 国产av精国产传媒| 国产成人自拍在线| 国产精品白丝久久av网站| 999视频在线| 视频福利一区| 亚洲国产一区自拍| 中文字幕影片免费在线观看| 免费人成黄页在线观看忧物| 日韩精品一区二区久久| 在线激情影院一区| 青青青视频在线免费观看| 欧美丰满老妇| 色94色欧美sute亚洲线路一ni| 国产精品视频1区| 中文字幕 国产| 极品少妇一区二区三区精品视频| 91精品视频专区| 不卡av中文字幕| 成人av午夜电影| 另类视频在线观看+1080p| 可以在线观看的黄色| 自拍一级黄色片| 国产精品中文字幕制服诱惑| 日韩黄色高清视频| 精品欧美一区二区久久久| 久久亚洲成人| 欧美大片免费观看| 久久亚洲精品国产| fc2ppv国产精品久久| av在线不卡网| 日韩资源av在线| 欧洲美女少妇精品| 亚洲一级二级三级| 中文亚洲视频在线| 无码人妻久久一区二区三区蜜桃| 精品综合久久88少妇激情| 亚洲欧美综合区自拍另类| 国产精品20p| 手机看片一区二区| 免费在线欧美视频| 99国产视频| 男女网站在线观看| 午夜久久中文| 成人一级片网址| 免费久久99精品国产自| 五月婷婷在线观看| 亚洲一区二区三区精品在线| 国产精品亚洲αv天堂无码| 国产精品黄色片| 亚洲成人a级网| 天海翼在线视频| 久久动漫亚洲| 操一操视频一区| 国产福利第一视频在线播放| 亚洲国产日韩一区二区| 精品久久久久久无| 99国内精品久久| 91九色精品| 日韩电影精品| 婷婷亚洲久悠悠色悠在线播放 | 农村末发育av片一区二区| 国产在线日韩精品| 国产69精品久久久久9| 一区二区视频免费| 久久久久久久久久电影| 被灌满精子的波多野结衣| 欧美激情不卡| 亚洲网站在线观看| 久久久久久久极品| 国产精品1024| 宅男在线精品国产免费观看| 345成人影院| 亚洲第一男人天堂| 男女做暖暖视频| 免费成人av在线| 欧美久久在线| 亚洲第一av| 亚洲精品ady| 久久久久久久福利| 国产在线麻豆精品观看| 一区二区成人国产精品| 成人免费视频观看| 亚洲性视频网站| 免费视频久久久| 久久综合久久综合久久| 少妇av一区二区三区无码| 66精品视频在线观看| 久久99精品久久久久久青青91| 9.1成人看片| 超黄网站在线观看| 日韩一区二区三区在线观看| 永久免费看片视频教学| 看片网站欧美日韩| 一区精品在线| 四虎视频在线精品免费网址| 中文字幕日韩综合av| 国产99久久久久久免费看| 久久久久9999亚洲精品| 免费av网址在线| 狠狠做深爱婷婷综合一区| 国产成+人+综合+亚洲欧美丁香花| 日本国产在线| 欧美在线一区二区| 国产精品久久久久久久av| 奇米影视一区二区三区| 亚洲一区二区四区| 国产精品高清一区二区| 色综合久久中文字幕综合网小说| а√天堂资源在线| 黄色一区二区在线| 亚洲在线观看网站| 免费国产在线视频| 亚洲一区二区三区四区在线免费观看 | 欧日韩精品视频| 国精产品久拍自产在线网站| 国产综合成人久久大片91| 黄色特一级视频| 国产毛片精品| 国产ts一区二区| 免费a级在线播放| 日韩天堂在线观看| 日韩精品在线观看免费| 中文av一区特黄| 少妇丰满尤物大尺度写真| 亚洲激情网站| 日韩欧美三级一区二区| 国产精久久一区二区| 久久久女人电视剧免费播放下载 | 成年人的黄色片| 欧美人体视频| 国产精品www色诱视频| 黄色在线免费网站| 欧美精品一区二区不卡| 大地资源第二页在线观看高清版| 极品av在线| 一区二区亚洲精品国产| 国产精品熟女久久久久久| 午夜精品123| 最新日韩免费视频| 日本一区二区免费在线观看| 制服.丝袜.亚洲.中文.综合懂色| 中文字幕亚洲乱码熟女1区2区| 久久黄色免费网站| 女仆av观看一区| 国产欧美精品在线播放| 1区2区在线| 综合国产在线视频| 少妇一级淫片免费看| 亚洲国产电影在线观看| 一区二区国产盗摄色噜噜| 女性隐私黄www网站视频| 韩国精品视频在线观看 | 在线精品视频小说1| 亚洲熟女www一区二区三区| 久久久久久麻豆| 在线播放av网址| 久久成人久久爱| av免费中文字幕| 韩国一区二区三区在线观看| 亚洲成人精品电影在线观看| 成人av地址| 91免费在线视频网站| 成人在线视频播放| 欧美亚洲国产精品| 欧美人与禽猛交乱配| 深夜福利一区二区| 男人的天堂av高清在线| 亚洲精品国产精品国自产在线| 99riav国产| 欧美精品免费视频| 中文字幕网址在线| 91国产精品成人| 国产高清中文字幕| 成人免费在线视频| 国产伦精品一区二区三区免费视频| 亚洲黄色中文字幕| 午夜精品在线视频| 任你弄在线视频免费观看| 久久最新资源网| 日本激情视频在线观看| 在线视频日韩精品| h视频在线播放| 尤物yw午夜国产精品视频| 黄色片在线免费观看| 国产网站欧美日韩免费精品在线观看 | 国产麻豆乱码精品一区二区三区 | 日韩一卡二卡三卡四卡| 国产美女主播在线| 欧洲精品一区| 亚洲精品一区久久久久久| 国产精品亚洲精品| 99热一区二区三区| 亚洲国产欧美91| 国产精品免费无码| 日本三级小视频| 免费黄网站在线观看| 午夜视频在线看| 国产在线|日韩| 亚洲人成精品久久久 | 另类一区二区| 精品久久久久久久久久久aⅴ| 首页国产欧美日韩丝袜| 亚洲美女免费在线| 在线不卡一区二区| 永久免费毛片在线播放不卡| 97久久天天综合色天天综合色hd | 欧美性猛交xxxx偷拍洗澡 | 茄子视频成人在线| 久久精品国产理论片免费| 欧美日韩亚洲国产成人| wwwxx日本| 免费av网站在线| 久久性爱视频网站| 黄页视频在线播放| 另类中文字幕网| 精品视频在线免费看| 国产一区二区激情| 91午夜在线播放| 欧美日韩在线一| 欧美成人一二三区| av中文字幕在线看| 不卡一区综合视频| 日韩在线免费观看视频| eeuss影院www在线播放| 久久久精品影院| 欧美精品videosex| 97视频在线观看免费| 巨胸喷奶水www久久久| 国产精品亚洲自拍| 精品国产第一国产综合精品| 国产一区在线观| 国内亚洲精品| 男女h黄动漫啪啪无遮挡软件| 国内精品久久久久久久97牛牛 | 僵尸世界大战2 在线播放| 91久久黄色| www.国产福利| 99久久精品免费看国产免费软件| 亚洲AV无码国产成人久久| 亚洲欧美日韩久久| 国产女同在线观看| 欧美久久一区二区| 国内精品久久久久久久久久久| 日韩av网站导航| 黄色免费在线看| 国产不卡一区二区在线播放| 四虎地址8848精品| 国产综合动作在线观看| 精品国产精品久久一区免费式| 国产精品久久成人免费观看| 国产日韩一区二区三区在线播放| 麻豆av免费在线| av成人动漫在线观看| 色偷偷www8888| 精品久久久久国产| 国产高清免费观看| 亚洲人成77777在线观看网| 精品视频在线一区二区| 日韩av日韩在线观看| 97se亚洲国产一区二区三区| 欧美亚洲爱爱另类综合| 在线日本成人| 一级特黄性色生活片| 成人av在线资源网| 国产传媒在线看| 欧美日韩色婷婷| 亚洲国产精品欧美久久| 日韩一区二区精品视频| 成人女同在线观看| 亚洲aⅴ男人的天堂在线观看| 国产在线观看91一区二区三区| 欧美深夜福利视频| 国产传媒欧美日韩成人| 国产精品suv一区二区88| 91福利视频在线| 午夜在线观看视频18| 欧美激情视频网站| 99精品中文字幕在线不卡| 日韩人妻精品一区二区三区| 日本中文字幕一区二区视频 | 日本综合视频| 免费在线观看一区二区| 亚洲激情av| 亚洲欧美高清在线| 亚洲黄色小说网站| 国产精品无码在线播放| 日韩视频免费在线| 精品无人乱码一区二区三区 | 在线视频欧美日韩| 中文在线免费二区三区| 俄罗斯精品一区二区| 66久久国产| 精品人妻一区二区三| 亚洲少妇30p| 在线免费看毛片| xvideos国产精品| 黄色成人在线视频| 欧美日韩国产三区| 男人的天堂亚洲| 国产精品无码午夜福利| 在线中文字幕不卡| 成人免费在线电影| 国产日韩专区在线| 亚洲最大av| 91人人澡人人爽| 欧美日韩午夜剧场| 免费在线国产| 国产精品高清在线| 999国产精品999久久久久久| 久热在线视频观看| 国产欧美精品国产国产专区| 中文久久久久久| 黄色片在线播放| 岛国一区二区在线观看| 午夜免费日韩视频| 亚洲天堂2024| 91精品丝袜国产高跟在线| 精品久久久视频| 奇米777四色影视在线看| 男人天堂综合| 99精品视频免费在线观看| 国产成人拍精品视频午夜网站 | 黄色亚洲大片免费在线观看| 日韩av中文字幕在线播放| 国产精品拍拍拍| 午夜精品久久久久久久久久蜜桃| 国产人成一区二区三区影院| 亚洲一区二区三区视频播放| 欧美激情一区二区三区免费观看| 精品久久美女| 欧美一区二区三区爱爱| 蜜臀久久99精品久久久酒店新书 | 91在线国产福利| 国产噜噜噜噜噜久久久久久久久 | 国产亚洲自拍一区| 91日本在线视频| 国产av一区二区三区| 黑人一区二区| 久久精品成人动漫| 国产jjizz一区二区三区视频|