成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

聊聊 JS 斷點的實現

開發 前端
JS 斷點的功能相信大家都用過,當我們設置一個斷點,然后代碼執行到這個斷點時,線程就會停住,然后我們點擊下一步的時候,又會再下一個斷點停住。那么這個停住到底意味著什么呢?下面這個圖是執行到一個斷點時 Node.js 的調用棧。

斷點的實現非常復雜,這里并不是說要長篇大論講解 JS 斷點在 V8 中是如何實現的,而是想從宏觀上聊一下斷點的實現。這個問題來源于最近和同事討論的關于 V8 Inspector 實現的一些事情。

JS 斷點的功能相信大家都用過,當我們設置一個斷點,然后代碼執行到這個斷點時,線程就會停住,然后我們點擊下一步的時候,又會再下一個斷點停住。那么這個停住到底意味著什么呢?下面這個圖是執行到一個斷點時 Node.js 的調用棧。

圖片

我們知道 V8 有一個調試協議,客戶端是和 V8 通過這個協議通信完成調試的,當 V8 收到客戶端的信息并且處理完之后,就會調用 runMessageLoopOnPause。runMessageLoopOnPause 是 V8 提供的一個約定的 API,當執行到 JS 斷點時就會調用,具體在 runMessageLoopOnPause 里做什么事情由 V8 的使用方實現。在看實現之前,先來思考一下,應該怎么處理。首先執行到了 JS 斷點,顯然線程就要進入停住的狀態,那么這個停住的狀態具體是指什么,應該怎么實現是一個最關鍵的問題。這個事件循環的實現有點類似,那就是當線程沒有任務處理的時候,它應該在做什么,輪詢顯然太不可思議了,那另一種就是基于訂閱 / 發布機制實現睡眠 / 喚醒,比如 Node.js 基于事件驅動模塊實現了睡眠 / 喚醒機制。類似的 Inspector 也是這樣實現,但是具體細節不一樣,因為如果情況不一樣,當 Node.js 處于事件循環的阻塞狀態時,任何注冊到事件驅動模塊的事件都可以喚醒 Node.js,但是斷點不一樣,當線程處于斷點時,除了信號外,一般的任務,比如文件 IO、網絡 IO 等,是不能也不應該能喚醒線程的,所以這里使用的是簡單的睡眠 / 喚醒方式,那就是條件變量。當線程阻塞于條件變量時,只有通過該條件變量才能喚醒線程。回到斷點的場景,那就是客戶端繼續執行時才能喚醒線程。

分析完之后,來看看 Node.js 的實現。

void runMessageLoopOnPause(int context_group_id) override {
waiting_for_resume_ = true;
runMessageLoop();
}

void runMessageLoop() {
if (running_nested_loop_)
return;

running_nested_loop_ = true;

while (shouldRunMessageLoop()) {
if (interface_) interface_->WaitForFrontendEvent();
env_->RunAndClearInterrupts();
}
running_nested_loop_ = false;
}

重點在 WaitForFrontendEvent。

bool MainThreadInterface::WaitForFrontendEvent() {
dispatching_messages_ = false;
// 任務隊列為空則阻塞
if (dispatching_message_queue_.empty()) {
Mutex::ScopedLock scoped_lock(requests_lock_);
while (requests_.empty()) incoming_message_cond_.Wait(scoped_lock);
}
return true;
}

我們假設這時候隊列為空,那么線程就會阻塞在條件變量 incoming_message_cond_ 中。接下來看看如聊聊第二個問題。線程這時候阻塞了,那么客戶端點擊執行下一步的時候,Node.js 還還怎么處理?這里就需要子線程幫忙了,所以 Node.js 中,和客戶端的數據通信是在子線程完成的,不講太多代碼和細節,直接看一個調用棧。

圖片

這是客戶端和 Node.js 子線程建立 websocket 連接成功后的調用棧,后續的數據通信也是類似。來看一下 Post。

void MainThreadInterface::Post(std::unique_ptr<Request> request) {
Mutex::ScopedLock scoped_lock(requests_lock_);
bool needs_notify = requests_.empty();
requests_.push_back(std::move(request));
if (needs_notify) {
std::weak_ptr<MainThreadInterface> weak_self {shared_from_this()};
agent_->env()->RequestInterrupt([weak_self](Environment*) {
if (auto iface = weak_self.lock()) iface->DispatchMessages();
});
}
incoming_message_cond_.Broadcast(scoped_lock);
}

這里看到了剛才熟悉的數據結構,Post 就是往主線程中插入一個任務,然后喚醒主線程。接著回到 runMessageLoop。

while (shouldRunMessageLoop()) {
if (interface_) interface_->WaitForFrontendEvent();
env_->RunAndClearInterrupts();
}

WaitForFrontendEvent 執行完畢后,接著執行 RunAndClearInterrupts,RunAndClearInterrupts 正是處理 RequestInterrupt 插入的任務的。剛才插入任務時我們看到插入了兩個任務 agent_->env()->RequestInterrupt 和 requests_.push_back(std::move(request)) ,RequestInterrupt 插入的任務中會調用 DispatchMessages,而 DispatchMessages 就是處理 requests_ 中的任務的。

void MainThreadInterface::DispatchMessages() {
dispatching_messages_ = true;
bool had_messages = false;
do {
if (dispatching_message_queue_.empty()) {
Mutex::ScopedLock scoped_lock(requests_lock_);
requests_.swap(dispatching_message_queue_);
}
had_messages = !dispatching_message_queue_.empty();
while (!dispatching_message_queue_.empty()) {
MessageQueue::value_type task;
std::swap(dispatching_message_queue_.front(), task);
dispatching_message_queue_.pop_front();

v8::SealHandleScope seal_handle_scope(agent_->env()->isolate());
task->Call(this);
}
} while (had_messages);
dispatching_messages_ = false;
}

執行任務的時候,具體做的事情就是把客戶端傳過來的數據投傳給 V8 Inspector,如果又執行到了一個斷點,那么繼續本文分析到這個邏輯,否則線程就可以繼續跑了。

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

2022-09-30 00:03:03

JS斷點線程

2017-04-19 10:25:01

JS斷點調試

2024-12-23 15:05:29

2021-09-26 05:06:04

Node.js模塊機制

2021-11-06 18:40:27

js底層模塊

2024-05-31 09:31:00

2022-02-18 08:26:12

TopK數組面試題

2021-07-14 14:05:24

Fragment項目結構

2022-03-26 16:51:27

Node.jstrace架構

2021-10-04 19:49:23

HTTP模塊No.js

2022-05-27 07:01:48

JSGIF總幀數

2017-03-06 16:51:52

Java泛型實現

2023-01-26 00:59:39

B-Treegolang度量衡

2025-04-24 00:00:00

Spring AI流式輸出AI 模型

2022-08-05 08:27:05

分布式系統線程并發

2022-05-13 09:05:37

JSObject無序

2017-08-08 08:45:44

前端文件斷點續傳

2023-05-26 08:24:17

短信渠道模型

2024-05-13 08:04:26

Vue.jsWeb應用程序

2024-04-07 08:23:01

JS隔離JavaScript
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久精品国产一区二区电影 | 性色av香蕉一区二区 | 网站黄色在线免费观看 | 欧美一级二级在线观看 | 伊人网综合 | аⅴ资源新版在线天堂 | 国产伊人精品 | 亚洲欧美国产精品久久 | 亚洲综合色网 | 欧州一区二区三区 | 国产激情视频在线免费观看 | 我想看一级黄色毛片 | 人人看人人射 | 在线观看亚洲欧美 | 亚洲成人精品在线观看 | 91社区视频 | 国产特级毛片aaaaaa喷潮 | 亚洲免费在线 | 一区二区三区免费 | 精品1区2区 | 偷拍自拍第一页 | 日韩在线不卡 | 亚洲一区二区中文字幕 | 成人在线精品视频 | 九九热免费看 | 欧美日韩精品区 | www.久久久久久久久久久久 | 国产不卡视频在线 | 欧美日韩亚洲视频 | 亚洲成人高清 | 久久久久久久久国产成人免费 | 一区二区精品在线 | 欧美精品一区在线发布 | 亚洲视频一区在线观看 | 色999视频 | 国产成人精品一区二区三 | 色综久久 | 国产一区二区三区久久久久久久久 | 欧美成人一区二区三区 | 国产日韩精品一区二区三区 | 国产精品一区一区三区 |