- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
为了简单起见,想象一下这个场景,我们有一台 2 位计算机,它有一对 2 位寄存器,称为 r1 和 r2,并且只适用于立即寻址。
假设位序列 00 意味着 add 到我们的 cpu。 01 表示将数据移动到 r1,10 表示将数据移动到 r2。
所以这台计算机有一个汇编语言和一个汇编器,其中的示例代码可以这样编写
mov r1,1
mov r2,2
add r1,r2
简单地说,当我将此代码组装成本地语言时,文件将类似于:
0101 1010 0001
上面的 12 位是 native 代码:
Put decimal 1 to R1, Put decimal 2 to R2, Add the data and store in R1.
所以这基本上就是编译代码的工作方式,对吧?
假设有人为此架构实现了 JVM。在 Java 中,我将编写如下代码:
int x = 1 + 2;
JVM 将如何准确地解释这段代码?我的意思是最终必须将相同的位模式传递给 cpu,不是吗?所有cpu都有许多可以理解和执行的指令,它们毕竟只是一些位。假设编译后的 Java 字节码看起来像这样:
1111 1100 1001
或其他什么..这是否意味着解释在执行时将此代码更改为 0101 1010 0001?如果是,它已经在 Native Code 中了,那为什么说 JIT 只在多次后才生效呢?如果它没有将它完全转换为 0101 1010 0001,那么它会做什么?它是如何让cpu做加法的?
也许我的假设有一些错误。
我知道解释很慢,编译代码更快但不可移植,虚拟机“解释”代码,但如何?我正在寻找“如何准确/技术解释”。欢迎任何指针(例如书籍或网页)而不是答案。
最佳答案
不幸的是,您描述的 CPU 架构过于受限,无法通过所有中间步骤来真正清楚地说明这一点。相反,我将编写伪 C 和伪 x86 汇编程序,希望以一种对 C 或 x86 非常熟悉的情况下清晰明了的方式。
编译后的 JVM 字节码可能如下所示:
ldc 0 # push first first constant (== 1)
ldc 1 # push the second constant (== 2)
iadd # pop two integers and push their sum
istore_0 # pop result and store in local variable
解释器将这些指令(二进制编码)放在一个数组中,以及一个引用当前指令的索引。它还有一个常量数组,一个用作堆栈的内存区域和一个用于局部变量的内存区域。然后解释器循环如下所示:
while (true) {
switch(instructions[pc]) {
case LDC:
sp += 1; // make space for constant
stack[sp] = constants[instructions[pc+1]];
pc += 2; // two-byte instruction
case IADD:
stack[sp-1] += stack[sp]; // add to first operand
sp -= 1; // pop other operand
pc += 1; // one-byte instruction
case ISTORE_0:
locals[0] = stack[sp];
sp -= 1; // pop
pc += 1; // one-byte instruction
// ... other cases ...
}
}
这个 C 代码被编译成机器代码并运行。如您所见,它是高度动态的:每次执行指令时,它都会检查每条字节码指令,并且所有值都通过堆栈(即 RAM)。
虽然实际加法本身可能发生在寄存器中,但加法周围的代码与 Java 到机器代码编译器发出的代码完全不同。以下是 C 编译器可能会将上述内容转换为 (pseudo-x86) 的摘录:
.ldc:
incl %esi # increment the variable pc, first half of pc += 2;
movb %ecx, program(%esi) # load byte after instruction
movl %eax, constants(,%ebx,4) # load constant from pool
incl %edi # increment sp
movl %eax, stack(,%edi,4) # write constant onto stack
incl %esi # other half of pc += 2
jmp .EndOfSwitch
.addi
movl %eax, stack(,%edi,4) # load first operand
decl %edi # sp -= 1;
addl stack(,%edi,4), %eax # add
incl %esi # pc += 1;
jmp .EndOfSwitch
您可以看到加法操作数来自内存而不是硬编码,即使对于 Java 程序而言它们是常量。那是因为对于解释器,它们不是恒定的。解释器被编译一次,然后必须能够执行各种程序,而无需生成专门的代码。
JIT 编译器的目的就是为了做到这一点:生成专门的代码。 JIT 可以分析堆栈用于传输数据的方式、程序中各种常量的实际值以及执行的计算顺序,以生成更有效地执行相同操作的代码。在我们的示例程序中,它会将局部变量 0 分配给一个寄存器,用将常量移入寄存器 (movl %eax, $1
) 代替对常量表的访问,并将堆栈访问重定向到正确的机器寄存器。忽略一些通常会进行的更多优化(复制传播、常量折叠和死代码消除),最终可能会得到如下代码:
movl %ebx, $1 # ldc 0
movl %ecx, $2 # ldc 1
movl %eax, %ebx # (1/2) addi
addl %eax, %ecx # (2/2) addi
# no istore_0, local variable 0 == %eax, so we're done
关于java - 解释器如何解释代码?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28135312/
我之前让 dll 注入(inject)器变得简单,但我有 Windows 7,我用 C# 和 C++ 做了它,它工作得很好!但是现在当我在 Windows 8 中尝试相同的代码时,它似乎没有以正确的方
我正在尝试制作一个名为 core-splitter 的元素,该元素在 1.0 中已弃用,因为它在我们的项目中起着关键作用。 如果您不知道 core-splitter 的作用,我可以提供一个简短的描述。
我有几个不同的蜘蛛,想一次运行所有它们。基于 this和 this ,我可以在同一个进程中运行多个蜘蛛。但是,我不知道如何设计一个信号系统来在所有蜘蛛都完成后停止 react 器。 我试过了: cra
有没有办法在达到特定条件时停止扭曲 react 器。例如,如果一个变量被设置为某个值,那么 react 器应该停止吗? 最佳答案 理想情况下,您不会将变量设置为一个值并停止 react 器,而是调用
https://code.angularjs.org/1.0.0rc9/angular-1.0.0rc9.js 上面的链接定义了外部js文件,我不知道Angular-1.0.0rc9.js的注入(in
我正在尝试运行一个函数并将服务注入(inject)其中。我认为这可以使用 $injector 轻松完成.所以我尝试了以下(简化示例): angular.injector().invoke( [ "$q
在 google Guice 中,我可以使用函数 createInjector 创建基于多个模块的注入(inject)器。 因为我使用 GWT.create 在 GoogleGin 中实例化注入(in
我在 ASP.NET Core 1.1 解决方案中使用配置绑定(bind)。基本上,我在“ConfigureServices Startup”部分中有一些用于绑定(bind)的简单代码,如下所示: s
我在 Spring MVC 中设置 initBinder 时遇到一些问题。我有一个 ModelAttribute,它有一个有时会显示的字段。 public class Model { privat
我正在尝试通过jquery post发布knockoutjs View 模型 var $form = $('#barcodeTemplate form'); var data = ko.toJS(vm
如何为包含多态对象集合的复杂模型编写自定义模型绑定(bind)程序? 我有下一个模型结构: public class CustomAttributeValueViewModel { publi
您好,我正在尝试实现我在 this article 中找到的扩展方法对于简单的注入(inject)器,因为它不支持开箱即用的特定构造函数的注册。 根据这篇文章,我需要用一个假的委托(delegate)
你好,我想自动注册我的依赖项。 我现在拥有的是: public interface IRepository where T : class public interface IFolderReposi
我正在使用 Jasmine 测试一些 Angular.js 代码。为此,我需要一个 Angular 注入(inject)器: var injector = angular.injector(['ng'
我正在使用 Matlab 代码生成器。不可能包含代码风格指南。这就是为什么我正在寻找一个工具来“ reshape ”、重命名和重新格式化生成的代码,根据我的: 功能横幅约定 文件横幅约定 命名约定 等
这个问题在这里已经有了答案: Where and why do I have to put the "template" and "typename" keywords? (8 个答案) 关闭 8
我开发了一种工具,可以更改某些程序的外观。为此,我需要在某些进程中注入(inject)一个 dll。 现在我基本上使用这个 approach .问题通常是人们无法注入(inject) dll,因为他们
我想使用 swing、spring 和 hibernate 编写一个 java 应用程序。 我想使用数据绑定(bind)器用 bean 的值填充 gui,并且我还希望它反射(reflect) gui
我有这段代码,当两个蜘蛛完成后,程序仍在运行。 #!C:\Python27\python.exe from twisted.internet import reactor from scrapy.cr
要点是 Spring Batch (v2) 测试框架具有带有 @Autowired 注释的 JobLauncherTestUtils.setJob。我们的测试套件有多个 Job 类提供者。因为这个类不
我是一名优秀的程序员,十分优秀!