最新在线看的黄网|伊人久久久久久久网站|日本a视频专区二|欧美A级无码毛片|有扫五av在线播放|好屌日aV在线播放|内射福利视频导航|极品少妇一区二区视频|无吗在线一区播放|性爱黄色视频不卡

JS斷點(diǎn)的實(shí)現(xiàn)的深入解讀

JS 斷點(diǎn)的功能相信大家都用過,當(dāng)我們?cè)O(shè)置一個(gè)斷點(diǎn),然后代碼執(zhí)行到這個(gè)斷點(diǎn)時(shí),線程就會(huì)停住,然后我們點(diǎn)擊下一步的時(shí)候,又會(huì)再下一個(gè)斷點(diǎn)停住。那么這個(gè)停住到底意味著什么呢?

斷點(diǎn)的實(shí)現(xiàn)非常復(fù)雜,這里并不是說要長(zhǎng)篇大論講解 JS 斷點(diǎn)在 V8 中是如何實(shí)現(xiàn)的,而是想從宏觀上聊一下斷點(diǎn)的實(shí)現(xiàn)。這個(gè)問題來源于最近和同事討論的關(guān)于 V8 Inspector 實(shí)現(xiàn)的一些事情。

JS 斷點(diǎn)的功能相信大家都用過,當(dāng)我們?cè)O(shè)置一個(gè)斷點(diǎn),然后代碼執(zhí)行到這個(gè)斷點(diǎn)時(shí),線程就會(huì)停住,然后我們點(diǎn)擊下一步的時(shí)候,又會(huì)再下一個(gè)斷點(diǎn)停住。那么這個(gè)停住到底意味著什么呢?下面這個(gè)圖是執(zhí)行到一個(gè)斷點(diǎn)時(shí) Node.js 的調(diào)用棧。

我們知道 V8 有一個(gè)調(diào)試協(xié)議,客戶端是和 V8 通過這個(gè)協(xié)議通信完成調(diào)試的,當(dāng) V8 收到客戶端的信息并且處理完之后,就會(huì)調(diào)用 runMessageLoopOnPause。runMessageLoopOnPause 是 V8 提供的一個(gè)約定的 API,當(dāng)執(zhí)行到 JS 斷點(diǎn)時(shí)就會(huì)調(diào)用,具體在 runMessageLoopOnPause 里做什么事情由 V8 的使用方實(shí)現(xiàn)。在看實(shí)現(xiàn)之前,先來思考一下,應(yīng)該怎么處理。首先執(zhí)行到了 JS 斷點(diǎn),顯然線程就要進(jìn)入停住的狀態(tài),那么這個(gè)停住的狀態(tài)具體是指什么,應(yīng)該怎么實(shí)現(xiàn)是一個(gè)最關(guān)鍵的問題。這個(gè)事件循環(huán)的實(shí)現(xiàn)有點(diǎn)類似,那就是當(dāng)線程沒有任務(wù)處理的時(shí)候,它應(yīng)該在做什么,輪詢顯然太不可思議了,那另一種就是基于訂閱 / 發(fā)布機(jī)制實(shí)現(xiàn)睡眠 / 喚醒,比如 Node.js 基于事件驅(qū)動(dòng)模塊實(shí)現(xiàn)了睡眠 / 喚醒機(jī)制。類似的 Inspector 也是這樣實(shí)現(xiàn),但是具體細(xì)節(jié)不一樣,因?yàn)槿绻闆r不一樣,當(dāng) Node.js 處于事件循環(huán)的阻塞狀態(tài)時(shí),任何注冊(cè)到事件驅(qū)動(dòng)模塊的事件都可以喚醒 Node.js,但是斷點(diǎn)不一樣,當(dāng)線程處于斷點(diǎn)時(shí),除了信號(hào)外,一般的任務(wù),比如文件 IO、網(wǎng)絡(luò) IO 等,是不能也不應(yīng)該能喚醒線程的,所以這里使用的是簡(jiǎn)單的睡眠 / 喚醒方式,那就是條件變量。當(dāng)線程阻塞于條件變量時(shí),只有通過該條件變量才能喚醒線程?;氐綌帱c(diǎn)的場(chǎng)景,那就是客戶端繼續(xù)執(zhí)行時(shí)才能喚醒線程。

分析完之后,來看看 Node.js 的實(shí)現(xiàn)。

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;
}

重點(diǎn)在 WaitForFrontendEvent。

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

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

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

void MainThreadInterface::Post(std::unique_ptr request) {
  Mutex::ScopedLock scoped_lock(requests_lock_);
  bool needs_notify = requests_.empty();
  requests_.push_back(std::move(request));
  if (needs_notify) {
    std::weak_ptr 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);
}

這里看到了剛才熟悉的數(shù)據(jù)結(jié)構(gòu),Post 就是往主線程中插入一個(gè)任務(wù),然后喚醒主線程。接著回到 runMessageLoop。

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

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

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;
}

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

阿里企業(yè)郵箱、網(wǎng)易企業(yè)郵箱、新網(wǎng)企業(yè)郵箱
【標(biāo)準(zhǔn)版】400元/年/5用戶/無限容量
【外貿(mào)版】500元/年/5用戶/無限容量
其它服務(wù):網(wǎng)站建設(shè)、企業(yè)郵箱、數(shù)字證書ssl、400電話、
聯(lián)系方式:電話:13714666846 微信同號(hào)

聲明:本站所有作品(圖文、音視頻)均由用戶自行上傳分享,或互聯(lián)網(wǎng)相關(guān)知識(shí)整合,僅供網(wǎng)友學(xué)習(xí)交流,若您的權(quán)利被侵害,請(qǐng)聯(lián)系 管理員 刪除。

本文鏈接:http://www.goalq.com.cn/article_32572.html