浅淡Ruby的文件加载与继承
文件加载
Ruby中load一个文件有四种方式,require
、require_relative
、load
、autoload
。
require和require_relative
require
是Ruby中最常见的加载一个文件的方式,如调用require 'rails'
时,会$LOAD_PATH
下寻找名为rails
的gem
包,然后将其lib
文件夹下的同名文件加载到Ruby虚拟机中来。多次require同一个包,只会加载一次。
require_relative
与require
类似,它只会在第一次调用时加载。区别是require_relative
的调用是相对路径。如当前文件夹下存在一个名为foo.rb
的文件时,调用的方式为require_relative 'foo'
。它不能调用$LOAD_PATH中的包
。
大约是Ruby2.0以后,require
也支持了相对路径的加载。比如上面的foo.rb
在当前目录时,通过require './foo'
也能达到require_relative 'foo'
的效果。
load
load
也是加载一个文件,它与require_relative
的区别是:
-
require_relative
多次加载同一文件时,只会加载一次;load
每一次调用都会重加载该文件。
autoload
autoload
是一种重要的加载方式,与require
的区别是require是即时加载
,autoload
的加载是懒加载
,即在需要它的时候才会被加载。autoload在某一个作用域内多次加载也只会被加载一次,因此不要以为它与load方法相似。
如果当前路径下有a.rb
、b.rb
两个文件:
class A
autoload :B, './b'
# 与 require './b' 等价,但autoload只有在调用 A::B 的时候才会去加载
end
autoload
第一个参数是类名的符号,第二个参数是加载的路径。它同时支持加载$LOAD_PATH
里的文件和相对路径
、绝对路径
的文件。
Rails
中重定义了autoload
方法,补充了一下path为空的情况下的常量寻找方法:
class Foo
require "active_support/all"
extend ActiveSupport::Autoload
autoload :B, "b"
end
class Bar
require "active_support/all"
extend ActiveSupport::Autoload
autoload :B
end
Bar
中的autoload
没有指定加载文件的路径,Rails
会自动生成加载路径bar/b
,而Foo
中的路径已指明,加载的路径是b
,因此两者加载的路径是不一样的。这是Rails
中autoload
与Ruby
中autoload
的区别。
变量与继承
关于对象模型,已经在元编程-对象模型篇讲过,此处不作详细说明。
类变量
类变量是以@@
开头命名的变量,在Ruby中,类变量是单例,在整个祖先链中是唯一的。
class A
@@a = 1
def self.a
@@a
end
def self.a=(a)
@@a = a
end
end
class B < A
end
B.a = 2
p A.a # 这里输出的值是2
代码中定义了一个类B
继承自A
,改变了B.a
的值,A.a
的值也跟着变化了。说明类变量是祖先链中唯一的。
这个专门问题被提出来讲,主要是前段时间面试的时候这个问题的理解上出了问题。之前以为两个类的self
不一样因此调用的不是同一个类变量对象。
由于类方法
和实例方法
都是可以被继承的,因此调用B.a
的时候,实质上B
中并不存在a
方法,因此调用的还是祖先链上游的A.a
方法,这与外部调用A.a
实际上效果一样。因此类变量是可以被继承的类所共享的。
类的实例变量
实例变量是以@
开头的变量,大部分情况下使用实例变量都是为类的实例
服务的。然而类本身,也是一个实例对象,因此类也可以有实例变量
。
class A
@a = 1
def self.a
@a
end
def self.a=(a)
@a = a
end
end
class B < A
end
p A.a # 1
p B.a # nil
B.a = 2
p A.a # 1
p B.a # 2
可以看到@a
对象是不共享的,A
和B
两个类都有自己独立的实例变量,因此修改任一个都不会影响另一个。