nodejs中怎么利用close實現(xiàn)事件循環(huán),相信很多沒有經(jīng)驗的人對此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個問題。
10年積累的成都網(wǎng)站設計、網(wǎng)站建設經(jīng)驗,可以快速應對客戶對網(wǎng)站的新想法和需求。提供各種問題對應的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡服務。我雖然不認識你,你也不認識我。但先網(wǎng)站制作后付款的網(wǎng)站建設流程,更有法庫免費網(wǎng)站建設讓你可以放心的選擇與我們合作。
close是nodejs每輪事件循環(huán)中最后的一個階段。我們看看怎么使用。我們知道對于一個handle,他的使用一般是init,start,stop。但是如果我們在stop一個handle之后,還有些事情需要處理怎么辦?這時候就可以使用close階段。close階段可以用來關閉一個handle,并且執(zhí)行一個回調(diào)。比如用于釋放動態(tài)申請的內(nèi)存。close階段的任務由uv_close產(chǎn)生。
void uv_close(uv_handle_t* handle, uv_close_cb close_cb) {
// 正在關閉,但是還沒執(zhí)行回調(diào)等后置操作
handle->flags |= UV_HANDLE_CLOSING;
handle->close_cb = close_cb;
switch (handle->type) {
case UV_PREPARE:
uv__prepare_close((uv_prepare_t*)handle);
break;
case UV_CHECK:
uv__check_close((uv_check_t*)handle);
break;
...
default:
assert(0);
}
uv__make_close_pending(handle);
}
uv_close設置回調(diào)和狀態(tài),然后根據(jù)handle類型調(diào)對應的close函數(shù),一般就是stop這個handle。比如prepare的close函數(shù)。
void uv__prepare_close(uv_prepare_t* handle) {
uv_prepare_stop(handle);
}
接著執(zhí)行uv__make_close_pending往close隊列追加節(jié)點。
// 頭插法插入closing隊列,在closing階段被執(zhí)行
void uv__make_close_pending(uv_handle_t* handle) {
handle->next_closing = handle->loop->closing_handles;
handle->loop->closing_handles = handle;
}
產(chǎn)生的節(jié)點在closing_handles隊列中保存,然后在close節(jié)點逐個處理。
// 執(zhí)行closing階段的的回調(diào)
static void uv__run_closing_handles(uv_loop_t* loop) {
uv_handle_t* p;
uv_handle_t* q;
p = loop->closing_handles;
loop->closing_handles = NULL;
while (p) {
q = p->next_closing;
uv__finish_close(p);
p = q;
}
}
// 執(zhí)行closing階段的回調(diào)
static void uv__finish_close(uv_handle_t* handle) {
handle->flags |= UV_HANDLE_CLOSED;
...
uv__handle_unref(handle);
QUEUE_REMOVE(&handle->handle_queue);
if (handle->close_cb) {
handle->close_cb(handle);
}
}
逐個執(zhí)行回調(diào),close和stop有一點不同的是,stop一個handle,他不會從事件循環(huán)中被移除,但是close一個handle,他會從事件循環(huán)的handle隊列中移除。
我們看一個使用了uv_close的例子(省略部分代碼)。
int uv_fs_poll_start(uv_fs_poll_t* handle,
uv_fs_poll_cb cb,
const char* path,
unsigned int interval) {
struct poll_ctx* ctx;
// 分配一塊堆內(nèi)存存上下文結(jié)構(gòu)體和path對應的字符串
ctx = uv__calloc(1, sizeof(*ctx) + len);
// 掛載上下文到handle
handle->poll_ctx = ctx;
}
uv_fs_poll_start是用于監(jiān)聽文件是否有改變的函數(shù)。他在handle里掛載了一個基于堆結(jié)構(gòu)體。當結(jié)束監(jiān)聽的時候,他需要釋放掉這塊內(nèi)存。
// 停止poll
int uv_fs_poll_stop(uv_fs_poll_t* handle) {
struct poll_ctx* ctx;
ctx = handle->poll_ctx;
handle->poll_ctx = NULL;
uv_close((uv_handle_t*)&ctx->timer_handle, timer_close_cb);
}
uv_fs_poll_stop通過uv_close函數(shù)關閉handle,傳的回調(diào)是timer_close_cb。
// 釋放上下文結(jié)構(gòu)體的內(nèi)存
static void timer_close_cb(uv_handle_t* handle) {
uv__free(container_of(handle, struct poll_ctx, timer_handle));
}
所以在close階段就會是否這塊內(nèi)存。
看完上述內(nèi)容,你們掌握nodejs中怎么利用close實現(xiàn)事件循環(huán)的方法了嗎?如果還想學到更多技能或想了解更多相關內(nèi)容,歡迎關注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!