深入V8引擎-初始化Platform之mac篇(1)

  本来寻思着写一篇"‘Hello‘ + ‘, World‘"是怎么从JS代码编译然后输出的,然而compile过程的复杂性远超我的想象,强上怕会走火入魔,还是老老实实先回家种田,找点咸鱼方法先写着。虽然说是咸鱼方法,但是V8任何一块拿出来都不简单,之前讲的Time模块说实话大概是属于源码里面幼儿园级别的,这次试试难一点的。

  V8的源码在本地编译完成后,会提供一个hello-world.cc的sample,里面有新手用户标准的初始化流程,如下。

int main(int argc, char* argv[]) {
  // Initialize V8.
  // 这个方法在mac不作为
  v8::V8::InitializeICUDefaultLocation(argv[0]);
  // 读取指定名称的配置文件 也不用鸟
  v8::V8::InitializeExternalStartupData(argv[0]);
  // 生成一个默认的platform对象
  std::unique_ptr<v8::Platform> platform = v8::platform::NewDefaultPlatform();
  // 初始化刚才的platform
  v8::V8::InitializePlatform(platform.get());
  // V8的初始化
  v8::V8::Initialize();

  // ...
}

  前两步不用去管,在入门阶段用不上。

  第三步是主要内容,探究生成的默认platform对象(当然也可以选择自己定制一个platform对象),这个类主要负责管理线程池、调用栈、线程task等一些杂活。

  这一篇不会去深入方法一步一步走,里面内容太过于杂乱,跳来跳去的,先整体介绍一下所有涉及的类,有一个初步的印象(建议深入阅读所有基类的英文注释,解释的很明白)。

Platform

  首先当然是核心类Platform,但这是一个基类,里面的大部分方法都是虚函数。

/**
 * V8 Platform abstraction layer.
 *
 * The embedder has to provide an implementation of this interface before
 * initializing the rest of V8.
 */
class Platform {};

  如果需要定制platform来初始化V8,需要继承这个类并实现那些方法。一般情况下当然可以V8默认提供的类,即DefaultPlatform。

class DefaultPlatform : public Platform {
  public:
    // 接受一个枚举值、一个TracingController类的构造函数
    explicit DefaultPlatform(
      IdleTaskSupport idle_task_support = IdleTaskSupport::kDisabled,
      std::unique_ptr<v8::TracingController> tracing_controller = {});
    ~DefaultPlatform() override;
    // 设置线程池大小
    void SetThreadPoolSize(int thread_pool_size);
    // 初始化线程池、管理线程任务相关的方法
    void EnsureBackgroundTaskRunnerInitialized();
  private:
    // 最大线程池数量 默认为8
    static const int kMaxThreadPoolSize;

    int thread_pool_size_;
    IdleTaskSupport idle_task_support_;
    // 线程任务启动器
    std::shared_ptr<DefaultWorkerThreadsTaskRunner> worker_threads_task_runner_;
    // 工具类
    std::unique_ptr<TracingController> tracing_controller_;
    std::unique_ptr<PageAllocator> page_allocator_;
    // 计数方法 用的是之前介绍的Time模块
    TimeFunction time_function_for_testing_;
};

/**
 * V8 Tracing controller.
 *
 * Can be implemented by an embedder to record trace events from V8.
 */
class TracingController {};

/**
 * A V8 memory page allocator.
 *
 * Can be implemented by an embedder to manage large host OS allocations.
 */
class PageAllocator {};

  只选了一些初始化相关的方法,其实内容远比这个要多。其中还定义了两个类似于Platform的基类变量,一个负责调用栈追踪,一个负责内存管理。

TaskRunner/Thread

  接下来是任务执行者、线程,因为这两者基本上成对出现,所以放一起来看。

// Thread
//
// Thread objects are used for creating and running threads. When the start()
// method is called the new thread starts running the run() method in the new
// thread. The Thread object should not be deallocated before the thread has
// terminated.

class V8_BASE_EXPORT Thread {
  public:
    // Start new thread by calling the Run() method on the new thread.
    void Start();
    // ...
};

  这是最基础的Thread,其中定义并实现了Start等常规方法,也有一些虚函数需要继承去重新实现,除此之外还有一些静态方法。默认情况下,V8自定义了一个线程类继承于基本Thread,位置十分的隐蔽,在默认TaskRunner的private里面。

/**
 * A TaskRunner allows scheduling of tasks. The TaskRunner may still be used to
 * post tasks after the isolate gets destructed, but these tasks may not get
 * executed anymore. All tasks posted to a given TaskRunner will be invoked in
 * sequence. Tasks can be posted from any thread.
 */
class TaskRunner {};

class DefaultWorkerThreadsTaskRunner : public TaskRunner {
  public:
    using TimeFunction = double (*)();
    DefaultWorkerThreadsTaskRunner(uint32_t thread_pool_size, TimeFunction time_function);
  private:
    class WorkerThread : public Thread {
    public:
      explicit WorkerThread(DefaultWorkerThreadsTaskRunner* runner);
      ~WorkerThread() override;

      // This thread attempts to get tasks in a loop from |runner_| and run them.
      void Run() override;
    private:
      DefaultWorkerThreadsTaskRunner* runner_;
    };
    // 获取下一个task
    std::unique_ptr<Task> GetNext();

    bool terminated_ = false;
    // task队列
    DelayedTaskQueue queue_;
    // 线程池
    std::vector<std::unique_ptr<WorkerThread>> thread_pool_;
    // 计数方法
    TimeFunction time_function_;
    std::atomic_int single_worker_thread_id_{0};
    uint32_t thread_pool_size_;
};

  这里顺便把TaskRunner相关的内容也一并放出来,大部分内容可以看命名。内部类的初始化参数类型是外部类,V8完全把Thread、TaskRunner两个类绑起来了。

Task

  这个类只是一个简单基类。

/**
 * A Task represents a unit of work.
 */
class Task {
 public:
  virtual ~Task() = default;
  // 所有的task需要继承这个类并实现Run方法
  virtual void Run() = 0;
};

  由于HelloWorld的sample并没有用到多线程,所以不存在Task类的实现,这里只能先关注概念。使用时,大概方法如下,写个伪代码演示下。

class userTask : public Task {
  public:
    void Run() {
      // do something...
    };
};

void handleTash() {
  // 新建一个task
  auto task = new userTask();
  // 加入队列
  queue_.push_back(task);
  // 唤醒线程
  thread_.signal();
  // 线程处理task
  auto task = queue_pop_back();
  task -> Run();
  // 线程等待唤醒
  thread_.wait();
}

  过程跟其实libuv的异步操作差不多,感觉编程的套路也就那样,看多了源码也就熟悉了。

  

  这一篇就介绍类的概念,了解后基本上V8中关于Platform的内容就差不多了。关于Thread、TaskRunner、Task三者的联系与运作,因为就学了2周C++,没去了解在C++中这些东西的实现与运用,所以暂时不在这里班门弄斧了。之前学Java的时候了解过线程,感觉无论是API的名字还是概念都差不多,有兴趣的可以自己去看看。

原文地址:https://www.cnblogs.com/QH-Jimmy/p/10964223.html

时间: 2024-10-22 21:54:35

深入V8引擎-初始化Platform之mac篇(1)的相关文章

V8 引擎是如何工作的?

V8 引擎是如何工作的? 本文翻译自:How the V8 engine works? ? V8是谷歌德国开发中心构建的一个JavaScript引擎.它是由C++编写的开源项目,同时被客户端(谷歌浏览器)和服务器端(Node.js)应用使用. ? V8最初是为了提高web浏览器中的JavaScript运行性能设计的.为了提升性能,V8将JavaScript代码翻译为更高效的机器语言,而不是使用解释程序.它通过实现一个JIT(Just-In-Time,即时)编译器来将JavaScript代码编译为

探究JS V8引擎下的“数组”底层实现

本文首发于 vivo互联网技术 微信公众号? 链接:https://mp.weixin.qq.com/s/np9Yoo02pEv9n_LCusZn3Q 作者:李超 JavaScript 中的数组有很多特性:存放不同类型元素.数组长度可变等等,这与数据结构中定义的数组结构或者C++.Java等语言中的数组不太一样,那么JS数组的这些特性底层是如何实现的呢,我们打开V8引擎的源码,从中寻找到了答案.V8中对数组做了一层封装,使其有两种实现方式:快数组和慢数组,快数组底层是连续内存,通过索引直接定位,

Chrome V8引擎系列随笔 (1):Math.Random()函数概览

先让大家来看一幅图,这幅图是V8引擎4.7版本和4.9版本Math.Random()函数的值的分布图,我可以这么理解 .从下图中,也许你会认为这是个二维码?其实这幅图告诉我们一个道理,第二张图的点的分布更加的密集,也就是说Math.Random()函数能表示的数字更多了,大家在.NET中肯定也用过GUID吧,至于GUID为什么会永不重复,大家有没有想过呢? 还是让我们先来看看官方怎么解释Math.Random()吧,它是返回了一个正数,这个正数介于0~1之间,以伪随机的方式在这个范围内波动.Ma

Node.js背后的V8引擎优化技术

Node.js的执行速度远超Ruby.Python等脚本语言,这背后都是V8引擎的功劳.本文将介绍如何编写高性能Node.js代码.V8是Chrome背后的JavaScript引擎,因此本文的相关优化经验也适用于基于Chrome浏览器的JavaScript引擎. V8优化技术概述 V8引擎在虚拟机与语言性能优化上做了很多工作.不过按照Lars Bak的说法,所有这些优化技术都不是他们创造的,只是在前人的基础上做的改进. 隐藏类(Hidden Class) 为了减少JavaScript中访问属性所

JavaScript工作机制:V8 引擎内部机制及如何编写优化代码的5个诀窍

概述 JavaScript引擎是一个执行JavaScript代码的程序或解释器.JavaScript引擎可以被实现为标准解释器,或者实现为以某种形式将JavaScript编译为字节码的即时编译器. 下面是实现了JavaScript引擎的一个热门项目列表: V8- 开源,由Google开发,用C++编写的 Rhino - 由Mozilla基金所管理,开源,完全用Java开发 SpiderMonkey-第一个JavaScript引擎,最早用在Netscape Navigator上,现在用在Firef

jquery jtemplates.js模板渲染引擎的详细用法第一篇

jquery jtemplates.js模板渲染引擎的详细用法第一篇 Author:ching Date:2016-06-29 jTemplates是一个基于JQuery的模板引擎插件,功能强大,有了他你就再不用为使用JS绑定数据时发愁了.后端语言使用php,asp.net,jsp等都不是问题,使用模板渲染可以很大程度上提高程序性能,使用异步获取数据,不用整个页面都回发,好处当然不仅仅是这些. 下载jtemplates,官网的文档写得非常的详细 打开官网:http://jtemplates.tp

Javascript的V8引擎研究

1.针对上下文的Snapshot技术 什么是上下文(Contexts)?实际是JS应用程序的运行环境,避免应用程序的修改相互影响,例如一个页面js修改内置对象方法toString,不应该影响到另外页面.chrome浏览器每个process只有一个V8引擎实例,浏览器中的每个窗口.iframe都对应一个上下文. ‍ V8启动时(在执行client js前),需要对全局上下文(第一个context)初始化,读取和解析自实现的内置JS代码(另一种技术,第2点),建立起function.array.st

V8引擎——详解

前言 JavaScript绝对是最火的编程语言之一,一直具有很大的用户群,随着在服务端的使用(NodeJs),更是爆发了极强的生命力.编程语言分为编译型语言和解释型语言两类,编译型语言在执行之前要先进行完全编译,而解释型语言一边编译一边执行,很明显解释型语言的执行速度是慢于编译型语言的,而JavaScript就是一种解释型脚本语言,支持动态类型.弱类型.基于原型的语言,内置支持类型.鉴于JavaScript都是在前端执行,而且需要及时响应用户,这就要求JavaScript可以快速的解析及执行.

Chrome V8系列--浅析Chrome V8引擎中的垃圾回收机制和内存泄露优化策略[转]

V8 实现了准确式 GC,GC 算法采用了分代式垃圾回收机制.因此,V8 将内存(堆)分为新生代和老生代两部分. 一.前言 V8的垃圾回收机制:JavaScript使用垃圾回收机制来自动管理内存.垃圾回收是一把双刃剑,其好处是可以大幅简化程序的内存管理代码,降低程序员的负担,减少因 长时间运转而带来的内存泄露问题. 但使用了垃圾回收即意味着程序员将无法掌控内存.ECMAScript没有暴露任何垃圾回收器的接口.我们无法强迫其进 行垃圾回收,更无法干预内存管理 内存管理问题:在浏览器中,Chrom