您的位置:首页 > 编程语言 > Java开发

SVN服务器端、客户端安装以及集成到eclipse的详细步骤(转)

2011-08-03 12:51 471 查看
    闲言碎语,以阐明ruby方法调用的原理。

    先决条件:

when a method is called, it is said that one is sending a message. The receiver is the object that the message is sent to.

当一个方法被调用,即说发送一个消息,这个消息的接收者即是方法的调用者。

例如:

o = Object.new

o.inspect

Object实例o调用了inspect方法,实际就是说给o发送一个调用inspect方法的消息。

 

    进入正题:

class A
end

class B < A
end

b = B.new
b.to_s #=>#<B:0x1cd3c0>

 执行过程:

1.沿b对象的klass指针找到B;(b是B的一个实例, b的klass指针指向B, klass指针通俗讲就是由实例指向类的指针,蕴含一种is_a?的概念,例如:b是B类的实例,则b的klass指针就指向B)

2.在B的m_tbl中查找是否存在to_s这个方法;(在类的内部存在一个m_tbl结构,实质是一个hash,它保存了该类的实例方法,注意:是实例方法;同时还存在一个iv_tbl的hash,保存该类的实例变量)

3.发现B内不存在to_s这个方法,则沿B的super指针找到A;(在类的内部存在一个super指针,指向其父类,外部调用就是B.superclass)

4.在A的m_tbl中仍然没有查找到to_s这个方法,于是沿A的super指针找到Object;(ruby 中的所有类都继承自Object)

5.Object包含to_s这个方法,因此该方法被调用(实际是调用原生方法 rb_any_to_s),返回类似"#<B:0x1cd3c0>"的结果。在这里,rb_any_to_s方法会检查原始消息的接收者的klass指针,来判断显示哪个类(是显示#<B:0x1cd3c0>,而不是#<A:0x1cd3c0>或#<Object:0x1cd3c0>)。原始消息的接收者是b,它是B类的实例,因此它的klass指向B,因此显示#<B:0x1cd3c0>。

ruby类的继承

    A继承Object, B继承A。 super chain: B ---> A ---> Object。这种由B到A,再到 Object的chain在ruby里称为lookup chain。

    这样当B的一个实例b调用某一方法时,会依次按照lookup chain进行查找。当类之间不仅有继承,而且有mixin的情况出现时,lookup chain就会有一些变化。如下:

module M
def test
puts "mod_method"
end
end

class A
def test
puts "cls_method"
end
end

class B < A
include M
end

b = B.new
b.test #=> mod_method

 

    这时,lookup chain: B ---> [M] ---> A ---> Object,这里在B到A之间插入了一个M,之所以加上[]标记,是为了区分它与lookup chain上其他成员的区别。实际上这个[M]是对module M的klass的一个代理,ruby里称为ICLASSes。[M]包含了指向Module M的m_tbl和iv_tbl的指针。

    当B的实例b调用test方法时,b会沿自己的klass指针找到B,然后在B中查找test方法,miss后直接沿super指针向上层找到[M],然后发现test方法并执行之。这也是为什么 b.test调用会返回mod_method而不是cls_method的原因。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: