深入V8引擎-默认Platform之mac篇(1)
又到了常规的堆砌代码凑文章字数环节,很多API我就直接贴官方的英文释义,个人翻译其实有时候并不是很准确,搞错了甚至会误导,还是尽量自己去理解。
首先看看入口方法。
std::unique_ptr<v8::Platform> platform = v8::platform::NewDefaultPlatform(); // int类型 thread_pool_size => 0
// 枚举值 idle_task_support => kDisabled
// 枚举值 in_process_stack_dumping => kDisabled
// 类 tracing_controller => NULL
std::unique_ptr<v8::Platform> NewDefaultPlatform(
int thread_pool_size, IdleTaskSupport idle_task_support,
InProcessStackDumping in_process_stack_dumping,
std::unique_ptr<v8::TracingController> tracing_controller) {
// 不会进这里
if (in_process_stack_dumping == InProcessStackDumping::kEnabled) {
v8::base::debug::EnableInProcessStackDumping();
} //
std::unique_ptr<DefaultPlatform> platform(new DefaultPlatform(idle_task_support, std::move(tracing_controller)));
//
platform->SetThreadPoolSize(thread_pool_size);
//
platform->EnsureBackgroundTaskRunnerInitialized();
return std::move(platform);
}
这里比较头疼的是我没找到NewDefaultPlatform方法是在哪里定义的,所以默认参数不知道是什么,只能打断点调试看变量值,已经在注释标注了。
默认参数情况下,那个if分支是不会进去的,所以无视,后面的三个语句都各自负责了一部门功能,分别是初始化默认Platform对象、设置线程池大小、启动工作线程,分块来看各部分源码。
第一条语句直接用new构造了一个DefaultPlatform对象,如下,加了一些标注。
//
DefaultPlatform::DefaultPlatform(
IdleTaskSupport idle_task_support,
std::unique_ptr<v8::TracingController> tracing_controller)
: thread_pool_size_(),
idle_task_support_(idle_task_support),
tracing_controller_(std::move(tracing_controller)),
// 1-1
page_allocator_(new v8::base::PageAllocator()),
time_function_for_testing_(nullptr) {
if (!tracing_controller_) {
// 1-2
tracing::TracingController* controller = new tracing::TracingController();
// 1-3
controller->Initialize(nullptr);
// 智能指针替换
tracing_controller_.reset(controller);
}
}
这是接受两个参数的构造函数,而且DefaultPlatform只有这一个构造函数。除了用给定的2个参数初始化属性,一些其他属性也用默认的参数初始化了,对于0、nullptr这种就不用管,其中比较特殊的是那个page_allocator初始化,上一篇给出的类声明是基类,在V8的命名空间有一个同名的实现类。
class PageAllocator : public ::v8::PageAllocator {
// ...
private:
const size_t allocate_page_size_;
const size_t commit_page_size_;
}
PageAllocator::PageAllocator()
: allocate_page_size_(base::OS::AllocatePageSize()),
commit_page_size_(base::OS::CommitPageSize()) {}
我也是服了V8,弄了个同名的类,第一次看楞了好久。构造函数调用的是OS命名空间的方法,这个命名空间是用来取一些系统参数,调用mac系统上<unistd.h>头文件的一些API,看一下allocate_page_size的初始化就明白了。
size_t OS::AllocatePageSize() {
return static_cast<size_t>(sysconf(_SC_PAGESIZE));
}
这里用了一个sysconf方法,在其他的很多地方也有使用,官方解释如下。
get configuration information at run time
当成NODE_ENV来理解就差不多了,也就是一个获取系统配置参数的API,对于PageAllocator的两个属性,官方的解释依次如下。
1、Size of a page in bytes. Must not be less than 1.2、memory page size
两个其实都是page size(内存页大小,关于Linux的内存模型我不太懂,后面有空再去了解),我本地测试了一下,都返回的4096。
对属性初始化完后,构造函数会继续走代码块里的语句,由于外部传进来的tracing_controller是个NULL,所以这里还需要手动new一个。
TracingController::TracingController() = default;
然而这个构造函数没啥好讲的,因为是默认构造函数,所以直接跳过了,后面的两步也没什么讲的,reset是智能指针的API,替换管理内容。
下面是第二条语句,从命名直接能看出来了,就是设置线程池的大小,方法也比较简单暴力了。
//
void DefaultPlatform::SetThreadPoolSize(int thread_pool_size) {
base::MutexGuard guard(&lock_);
DCHECK_GE(thread_pool_size, );
if (thread_pool_size < ) {
// The number of processors currently online (available) => 4
thread_pool_size = base::SysInfo::NumberOfProcessors() - ;
}
// max(min(8, size), 1)
thread_pool_size_ = std::max(std::min(thread_pool_size, kMaxThreadPoolSize), );
}
这里实际上也是调用了一个类似于上面的sysconf来获取系统参数,返回的系统处理器数量,由于其中一个要用来作为主线程,所以可用的线程池数量要减一,简单处理一下返回一个size。
第三条语句内容相当的麻烦,看得我脑子疼,语义上理解就是保证后台线程runner的初始化运行。
说得简单,由于之前的操作只是初始化了一个空白platform类,算了算线程池的大小,所以剩下的所有实际操作都在这里。大概包含了初始化线程池、生成线程同时分配任务、管理task队列、启动线程等一系列操作,其中的思想倒并不复杂,实际上跟libuv的异步原理相差无几,但是深入源码的每一步还是挺恶心的,下一篇再来搞。
深入V8引擎-默认Platform之mac篇(1)的更多相关文章
- 深入V8引擎-默认Platform之mac篇(2)
先说结论,V8引擎在默认Platform中初始化的这个线程是用于处理类似于setTimeout的延时任务. 另外附一些图,包括继承树.关键属性归属.纯逻辑工作流程,对代码木得兴趣的看完图可以X掉了. ...
- 深入V8引擎-初始化默认Platform
本来寻思着写一篇"'Hello' + ', World'"是怎么从JS代码编译然后输出的,然而compile过程的复杂性远超我的想象,强上怕会走火入魔,还是老老实实先回家种田,找点 ...
- 深入V8引擎-引擎内部类管理解析
v8的初始化三部曲,前面花了三篇解决了第一步,由于只是生成了一个对象,第二步就是将其嵌入v8中,先看一下三个步骤. // 生成默认Platform对象 std::unique_ptr<v8::P ...
- V8引擎嵌入指南
如果已读过V8编程入门那你已经熟悉了如句柄(handle).作用域(scope)和上下文(context)之类的关键概念,以及如何将V8引擎作为一个独立的虚拟机来使用.本文将进一步讨论这些概念,并介绍 ...
- 浅谈V8引擎中的垃圾回收机制
最近在看<深入浅出nodejs>关于V8垃圾回收机制的章节,转自:http://blog.segmentfault.com/skyinlayer/1190000000440270 这篇文章 ...
- V8引擎——详解
前言 JavaScript绝对是最火的编程语言之一,一直具有很大的用户群,随着在服务端的使用(NodeJs),更是爆发了极强的生命力.编程语言分为编译型语言和解释型语言两类,编译型语言在执行之前要先进 ...
- v8引擎详解
引用网址: https://blog.csdn.net/swimming_in_it_/article/details/78869549 前言 JavaScript绝对是最火的编程语言之一,一直具有很 ...
- 一文搞懂V8引擎的垃圾回收
引言 作为目前最流行的JavaScript引擎,V8引擎从出现的那一刻起便广泛受到人们的关注,我们知道,JavaScript可以高效地运行在浏览器和Nodejs这两大宿主环境中,也是因为背后有强大的V ...
- 探究JS V8引擎下的“数组”底层实现
本文首发于 vivo互联网技术 微信公众号 链接:https://mp.weixin.qq.com/s/np9Yoo02pEv9n_LCusZn3Q作者:李超 JavaScript 中的数组有很多特性 ...
随机推荐
- JavaScript深入浅出第5课:Chrome是如何成功的?
摘要: Chrome改变世界. <JavaScript深入浅出>系列: JavaScript深入浅出第1课:箭头函数中的this究竟是什么鬼? JavaScript深入浅出第2课:函数是一 ...
- SpringBoot的学习二:整合Redis,JPA,Mybatis
Redis介绍: 是一个开源的使用ANSI C语言编写.支持网络.可基于内存亦可持久化的日志型.Key-Value数据库,并提供多种语言的API 特性: Redis 与其他 key - value 缓 ...
- mysql常用配置注意项与sql优化
建立数据库: 建立数据库时编码字符集采用utf8 排序规则: 后缀"_cs"或者"_ci"意思是区分大小写和不区分大小写(Case Sensitive & ...
- Rust中的迭代器
和闭包一样,练代码 struct Counter { count: u32, } impl Counter { fn new() -> Counter { Counter {count: } } ...
- Ubuntu下安装Rabbitmq和golang环境
安装及配置Rabbitmq 1. 安装: sudo apt-get install rabbitmq-server 2. 启动web管理插件 sudo rabbitmq-plugins enable ...
- CMS垃圾收集器深入详解
上一次[https://www.cnblogs.com/webor2006/p/11048407.html]对安全点和安全区进行了理论化的了解,接下来继续对CMS进行其它理论的了解,还是纯理论!!坚持 ...
- ansible 软件相关模块,剧本
软件相关模块 yum rpm和yum的区别 rpm:redhat package manager yum 可以解决依赖关系 yum 源配置 使用yum下载时需要先下载epel [epel] name= ...
- Nacos 知识点
Nacos 名字的由来(取红色的英文字符): Dynamic Naming and Configuration Service Nacos 是 Spring Cloud Alibaba 的一个组件,详 ...
- SQL-select常用语句
1.全套装备 select [select选项] 字段列表[字段别名]/* from 数据源[where 条件子句] [group by条件子句] [having 子句] [order by 子句] ...
- Python进阶-VII 内置函数
一.内置函数引入 我们已经了解的有; print() input() range() next() dir() str() int() list() set() tuple() dict() he ...