命名空间
class Text *** end
自己写了一个名为Text的类,但运行时报错:Text is not a class。调查发现,原来Action Mailer中有个模块,它也叫Text。由于Text已经被定义为模块,所以Ruby不能把它当做一个类。
module Text *** end
研究发现,使用命名空间可以轻松解决我的Text类与Action Mailer中Text类的命名冲突的问题。只需要将自己写的类放到一个命名空间里即可,使用My_Module::Text来进行使用。
module My_Module class Text *** end
接收者
接收者就是你调用方法所在的对象。例如,在my_string.reverse()语句中,my_string就是接收者。
祖先链
祖先链的对某一个对象,找到该对象的类,然后找到该类的超类,然后在找到超类的超类,依此类推,一直找到BasicObject类。经历的类路径叫做该类的祖先链(祖先链可能包括模块)。
在上图中,调用my_method方法时,Ruby会先向右找到它的MySubclass类,由于这个类中没有my_method方法,Ruby就会向上查找父类,一直找到BasicObject类。这种查找规则称为"向右一步,再向上"。通过ancestors方法获得类的祖先链:
细化
上一篇中打开类时说到猴子补丁,会导致类似Sting类的方法被覆盖或修改,导致所有调用原来Sting类中方法结果异常。从Ruby2.0开始,可以使用细化来解决这个问题。首先,需要定义一个模块,然后在这个模块中调用refine方法:
这段代码为String类细化了一个to_alphanumeric方法,与打开不同,细化在默认情况下并不生效,如果直接调用String#to_alphanumeric方法会报错。为了让这个方法生效,必须调用using方法:
从调用using方法的这一刻起,所有Ruby代码都知道了这个变化。例如,可以在一个模块内部使用using,这样,细化作用范围只在该模块有效。
细化和打开类作用相似,区别在于细化不是全局性的,细化只作用在你希望生效的地方。细化只在两种场合有效:1,refine代码块内部,2,从使用using语句开始,到该模块结束(如果在模块内部调用)。
网友评论