首页 > 关于动态语言运行机制的一些疑问

关于动态语言运行机制的一些疑问

今天看到了这个,上面说ruby这种动态脚本式从代码的第一行开始顺序执行的,然后又想到了python,所以又看到了这篇,然后心生了疑问,python和ruby中如果要在当前的函数中调用其他的函数的话
* 当前函数的名字为func1
* 被调用的函数的名字为func2

那么func2可以定义在func1之前也可以定义在func1之后,那按照第一篇博文中说的顺序执行,那如果func2定义在func1之后的话当执行func1的时候就不是找不到func2的定义了么,像python的话还会生成一个中间的pyc或者pyo还可以理解,可是ruby在解释的时候没有生成任何的中间文件,ruby虚拟机是如何做到寻找之前没有执行过的函数的,也就是在func2定义在func1之后的时候python和ruby是如何做到在func1中执行func2而不出错的


python我不清楚,在Ruby里面,def是一个表达式,返回值是nil(Ruby 2.1以前)或者定义的方法的符号(Ruby2.1开始)
def的效果是给对应的对象或者类添加了一个方法。

换句话说,def必须执行才会有对应的方法。

比如

  p(def add(a,b) 
     a+b
   end)

应该打印出nil或者:add。
同时在对应的地方定义了一个名为add的方法用来算加法。这之后对于add的查找才会考虑这个‘定义’表达式的效果。


解释器会先扫描整个文件,找语法错的,扫描一遍函数名不就都知道了嘛


不需要中间文件,func1, func2的定义保存在内存中即可。

以PHP为例,在内存里有一个HashTable,专门用于保存user difined function,当PHP Engine解析(Parse)到function func2()时,就往这个HashTable里插入一条记录,Key是func2,Value是func2的内存指针地址等信息。

当func1.php调用(Call)func2时,PHP Engine就去这个HashTable里查找func2在不在,如果在就继续调用,如果不在就报错:call to undefined function func2.

这和编译型语言不一样,编译型语言是在编译阶段(Compiling)就会判断func2在不在,如果func2不存在,编译器就会报错。而动态语言(解释性语言)是到了运行阶段(runtime)才会报错的。之所以存在这样的区别,是为了达到动态语言所谓的动态灵活性,func2可能并不是在hard code中写死的一个函数名,而是一个变量(比如根据if else算出来的、End User通过HTTP传过来的,等等),例如:

<?php
$funcName = $_POST["action"] . "Article";//假设这里$_POST["action']=del
$funcName(1001);//则这里实际调了delArticle(1001)

那么存在中间文件的意义是什么呢,中间文件可以把一些语法解析的结果缓存起来,下次执行的时候,如果源代码没有改动,就直接执行中间文件里的缓存指令了,这样可以极大地提升性能。


关于Ruby的链接打不开。python不了解。关于ruby提供一个简单的例子。

lambda { 
    def fun1
        puts 'fun1'
        fun2
    end

    def fun2
        puts 'fun2'
    end

    fun1
}.call
#运行结果
#fun1
#fun2

但是如果再定义fun2之前调用fun2

lambda { 
    def fun1
        puts 'fun1'
        fun2
    end
    fun1
    def fun2
        puts 'fun2'
    end

}.call

会给出如下错误
fun1': undefined local variable or methodfun2' for main:Object (NameError)
而且根据提示的行数,是再调用fun1的时候出错的。也就是说ruby是再定义方法的时候并不会去检查变量是不是真正存在,只有在运行fun1时才会去真正执行这段代码。这也是动态语言的特性。也只有这样,也才能保证随时可以重新打开方法,重写方法。

【热门文章】
【热门文章】