文件加載
Ruby中l(wèi)oad一個(gè)文件有四種方式,require
、require_relative
甥绿、load
、autoload
则披。
require和require_relative
require
是Ruby中最常見(jiàn)的加載一個(gè)文件的方式共缕,如調(diào)用require 'rails'
時(shí),會(huì)$LOAD_PATH
下尋找名為rails
的gem
包士复,然后將其lib
文件夾下的同名文件加載到Ruby虛擬機(jī)中來(lái)图谷。多次require同一個(gè)包,只會(huì)加載一次阱洪。
require_relative
與require
類似便贵,它只會(huì)在第一次調(diào)用時(shí)加載。區(qū)別是require_relative
的調(diào)用是相對(duì)路徑冗荸。如當(dāng)前文件夾下存在一個(gè)名為foo.rb
的文件時(shí)承璃,調(diào)用的方式為require_relative 'foo'
。它不能調(diào)用$LOAD_PATH中的包
蚌本。
大約是Ruby2.0以后盔粹,require
也支持了相對(duì)路徑的加載隘梨。比如上面的foo.rb
在當(dāng)前目錄時(shí),通過(guò)require './foo'
也能達(dá)到require_relative 'foo'
的效果舷嗡。
load
load
也是加載一個(gè)文件轴猎,它與require_relative
的區(qū)別是:
-
require_relative
多次加載同一文件時(shí),只會(huì)加載一次进萄;load
每一次調(diào)用都會(huì)重加載該文件捻脖。
autoload
autoload
是一種重要的加載方式,與require
的區(qū)別是require是即時(shí)加載
垮斯,autoload
的加載是懶加載
郎仆,即在需要它的時(shí)候才會(huì)被加載。autoload在某一個(gè)作用域內(nèi)多次加載也只會(huì)被加載一次兜蠕,因此不要以為它與load方法相似扰肌。
如果當(dāng)前路徑下有a.rb
、b.rb
兩個(gè)文件:
class A
autoload :B, './b'
# 與 require './b' 等價(jià)熊杨,但autoload只有在調(diào)用 A::B 的時(shí)候才會(huì)去加載
end
autoload
第一個(gè)參數(shù)是類名的符號(hào)曙旭,第二個(gè)參數(shù)是加載的路徑。它同時(shí)支持加載$LOAD_PATH
里的文件和相對(duì)路徑
晶府、絕對(duì)路徑
的文件桂躏。
Rails
中重定義了autoload
方法,補(bǔ)充了一下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
沒(méi)有指定加載文件的路徑川陆,Rails
會(huì)自動(dòng)生成加載路徑bar/b
剂习,而Foo
中的路徑已指明,加載的路徑是b
较沪,因此兩者加載的路徑是不一樣的鳞绕。這是Rails
中autoload
與Ruby
中autoload
的區(qū)別。
變量與繼承
關(guān)于對(duì)象模型尸曼,已經(jīng)在元編程-對(duì)象模型篇講過(guò)们何,此處不作詳細(xì)說(shuō)明。
類變量
類變量是以@@
開(kāi)頭命名的變量控轿,在Ruby中冤竹,類變量是單例,在整個(gè)祖先鏈中是唯一的茬射。
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
代碼中定義了一個(gè)類B
繼承自A
,改變了B.a
的值鹦蠕,A.a
的值也跟著變化了。說(shuō)明類變量是祖先鏈中唯一的在抛。
這個(gè)專門問(wèn)題被提出來(lái)講片部,主要是前段時(shí)間面試的時(shí)候這個(gè)問(wèn)題的理解上出了問(wèn)題。之前以為兩個(gè)類的self
不一樣因此調(diào)用的不是同一個(gè)類變量對(duì)象霜定。
由于類方法
和實(shí)例方法
都是可以被繼承的档悠,因此調(diào)用B.a
的時(shí)候,實(shí)質(zhì)上B
中并不存在a
方法望浩,因此調(diào)用的還是祖先鏈上游的A.a
方法辖所,這與外部調(diào)用A.a
實(shí)際上效果一樣。因此類變量是可以被繼承的類所共享的磨德。
類的實(shí)例變量
實(shí)例變量是以@
開(kāi)頭的變量缘回,大部分情況下使用實(shí)例變量都是為類的實(shí)例
服務(wù)的。然而類本身典挑,也是一個(gè)實(shí)例對(duì)象酥宴,因此類也可以有實(shí)例變量
。
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
對(duì)象是不共享的您觉,A
和B
兩個(gè)類都有自己獨(dú)立的實(shí)例變量拙寡,因此修改任一個(gè)都不會(huì)影響另一個(gè)。