我听说过类似“C运行时”,“Visual c++ 2008运行时”,“。NET公共语言运行时”等。

什么是“运行时”? 它是用什么做的? 它如何与我的代码交互?或者更准确地说,我的代码是如何被它控制的?

在Linux上编写汇编语言时,我可以使用INT指令进行系统调用。那么,运行时只是一堆将低级函数包装成更抽象和高级函数的预制函数吗?但这看起来更像是库的定义,而不是运行时的定义吗?

“运行时”和“运行时库”是两个不同的东西吗?

加1

这些天,我在想也许运行时与所谓的虚拟机(如JVM)有一些共同之处。下面这句话引出了这样的想法:

这个编译过程非常复杂,很难分解 有几个抽象层,这些抽象层通常包括三个 翻译器:编译器、虚拟机实现和 汇编程序。——《计算系统的要素》(导论, 通往硬件领域的道路)

加上2

《专家C编程:深层C语言秘密》一书。第6章运行时数据结构是对这个问题的有用参考。

加3 - 2021年2月28日上午7:31

以下是我对处理器设计有所了解后的一些看法。整个计算机就是多个层次的抽象。它从基本的晶体管一直到运行的程序。对于任何抽象级别N,它的运行时间是直接的抽象级别N-1,在它下面。是上帝给了我们抽象的0级。


当前回答

马特·鲍尔答对了。我会举例说明。

考虑运行一个用Turbo-Borland C/ c++(1991年的3.1版本)编译器编译的程序,并让它在32位版本的windows(如Win 98/2000等)下运行。

它是一个16位的编译器。你会看到你所有的程序都有16位指针。为什么你的操作系统是32位的?因为你的编译器已经设置了16位的执行环境,并且32位版本的操作系统支持它。

通常所称的JRE (Java运行时环境)为Java程序提供了执行它可能需要的所有资源。

实际上,运行时环境是虚拟机思想的大脑产物。虚拟机实现硬件和程序可能需要执行的内容之间的原始接口。运行时环境采用这些接口,并将它们呈现给程序员使用。编译器开发人员需要这些工具为其程序提供执行环境。

其他回答

运行时在某种程度上与设计时和编译时/链接时相反。从历史上看,它来自缓慢的大型机环境,在那里机器时间非常昂贵。

马特·鲍尔答对了。我会举例说明。

考虑运行一个用Turbo-Borland C/ c++(1991年的3.1版本)编译器编译的程序,并让它在32位版本的windows(如Win 98/2000等)下运行。

它是一个16位的编译器。你会看到你所有的程序都有16位指针。为什么你的操作系统是32位的?因为你的编译器已经设置了16位的执行环境,并且32位版本的操作系统支持它。

通常所称的JRE (Java运行时环境)为Java程序提供了执行它可能需要的所有资源。

实际上,运行时环境是虚拟机思想的大脑产物。虚拟机实现硬件和程序可能需要执行的内容之间的原始接口。运行时环境采用这些接口,并将它们呈现给程序员使用。编译器开发人员需要这些工具为其程序提供执行环境。

运行时可以表示程序生命的当前阶段(运行时/编译时间/加载时间/链接时间) 或者它可能意味着一个运行时库,它形成了与执行环境交互的基本低级操作。 或者它可能意味着运行时系统,这与执行环境相同。

在C程序中,运行时是建立堆栈、堆等的代码,这是C环境所期望的要求。它基本上建立了语言所承诺的环境。(它可以有一个运行时库组件crt0。lib或类似的C)

运行时或执行环境是语言实现的一部分,它在运行时执行代码;实现的编译时部分在C标准中称为翻译环境。

例子:

Java运行时由虚拟机和标准库组成 一个普通的C运行时由加载器(它是操作系统的一部分)和运行时库组成,运行时库实现了C语言中没有被编译器内置到可执行文件中的部分;在托管环境中,这包括标准库的大部分

我对其他答案不太感兴趣;它们对我来说太模糊太抽象了。我更喜欢讲故事。下面是我试图给出的一个更好的答案。

一个BASIC的例子

假设现在是1985年,你在Apple II上写了一个简短的BASIC程序:

] 10 PRINT "HELLO WORLD!"
] 20 GOTO 10

到目前为止,您的程序只是源代码。它没有运行,我们可以说它不涉及“运行时”。

但现在我运行它:

] RUN

它是如何运行的?它如何知道如何将字符串参数从PRINT发送到物理屏幕?我当然没有在代码中提供任何系统信息,而且PRINT本身也不知道关于系统的任何信息。

相反,RUN实际上是一个程序本身——它的代码告诉它如何解析我的代码,如何执行它,以及如何向计算机的操作系统发送任何相关的请求。RUN程序提供了“运行时”环境,作为操作系统和源代码之间的一层。操作系统本身作为这个“运行时”的一部分,但当我们谈论像RUN程序这样的“运行时”时,我们通常并不打算包括它。

编译和运行时的类型

编译二进制语言

在某些语言中,必须先编译源代码才能运行。有些语言将你的代码编译成机器语言——它可以由你的操作系统直接运行。这种编译后的代码通常被称为“二进制”(即使其他类型的文件也是二进制的:)。

在这种情况下,仍然涉及到一个最小的“运行时”——但是这个运行时是由操作系统本身提供的。编译步骤意味着在代码运行之前检测到许多可能导致程序崩溃的语句。

C is one such language; when you run a C program, it's totally able to send illegal requests to the operating system (like, "give me control of all of the memory on the computer, and erase it all"). If an illegal request is hit, usually the OS will just kill your program and not tell you why, and dump the contents of that program's memory at the time it was killed to a .dump file that's pretty hard to make sense of. But sometimes your code has a command that is a very bad idea, but the OS doesn't consider it illegal, like "erase a random bit of memory this program is using"; that can cause super weird problems that are hard to get to the bottom of.

字节码的语言

其他语言(如Java、Python)将代码编译成操作系统不能直接读取的语言,但特定的运行时程序可以读取编译后的代码。这种编译后的代码通常被称为“字节码”。

The more elaborate this runtime program is, the more extra stuff it can do on the side that your code did not include (even in the libraries you use) -- for instance, the Java runtime environment ("JRE") and Python runtime environment can keep track of memory assignments that are no longer needed, and tell the operating system it's safe to reuse that memory for something else, and it can catch situations where your code would try to send an illegal request to the operating system, and instead exit with a readable error.

所有这些开销使它们比编译的二进制语言慢,但它使运行时强大而灵活;在某些情况下,它甚至可以在开始运行后引入其他代码,而不必重新开始。编译步骤意味着在代码运行之前检测到许多会导致程序崩溃的语句;强大的运行时可以防止你的代码做一些愚蠢的事情(例如,你不能“擦除这个程序正在使用的随机内存”)。

脚本语言

Still other languages don't precompile your code at all; the runtime does all of the work of reading your code line by line, interpreting it and executing it. This makes them even slower than "bytecode" languages, but also even more flexible; in some cases, you can even fiddle with your source code as it runs! Though it also means that you can have a totally illegal statement in your code, and it could sit there in your production code without drawing attention, until one day it is run and causes a crash.

这些通常被称为“脚本”语言;它们包括Javascript、Perl和PHP。其中一些提供了你可以选择编译代码以提高其速度的情况(例如,Javascript的WebAssembly项目)。所以Javascript可以让网站上的用户看到正在运行的确切代码,因为他们的浏览器提供了运行时。

这种灵活性还允许在运行时环境中进行创新,如node.js,它既是一个代码库,也是一个运行时环境,可以将Javascript代码作为服务器运行,这涉及到与试图在浏览器上运行相同代码的行为非常不同。