打开类
定义:可以重新打开已经存在的类并对之进行动态修改,即使像String或者Array这样标准库的类也不例外。
问题:如果事先没有调查清楚就为某个类添加新方法,有可能无意中覆盖了类中原有的方法。有些人不喜欢这种鲁莽地修改类的方式,他们给它起个不好听的名字-猴子补丁。
对象的真相
实例变量:对象包含实例变量,可以通过Object#instance_variables方法查看,上面例子中,obj对象只有一个实例变量。
Ruby中对象的类和它的实例变量没有关系,当给实例变量赋值时突然出现。上面例子中如果没有调用obj.my_method方法,obj对象就没有任何实例变量。
方法:除了实例变量外,对象还有方法。可以通过调用Object#methods方法获得一个对象的方法列表。但对象并没有真正存放一组方法,一个对象仅包含它的实例变量以及一个对自身类的引用。方法存在于类中,非对象中。
注:规范方法的口语表达方式。obj对象有my_method方法,所以可以这么调用obj.my_method。但是,不能说MyClass有一个my_method方法,会引起误会,如果这样说意味着它是一个类方法,可以这样调用MyClass.my_method。所以为了消除起义,应该说my_method是MyClass类的一个实例方法(而不是简单的叫做方法),这意味着这个方法定义在MyClass类中,需要定义一个MyClass的实例才能调用它。
总结:一个对象的实例变量存在于对象本身之中,而一个对象的方法存在于对象自身的类中。这就是同一个类的对象共享同样的方法,但不共享实例变量的原因。
类的真相
Ruby对象模型最重要的知识:类本身也是对象。所以类像其他对象一样,也有自己的类,叫做Class。即类的方法就是Class的实例方法(上面特意规范了口语表达方式的)。
new方法是创建对象的,allocate是new方法的支撑方法,使用较少,superclass方法是获取父类。
Array类继承自Object,即"数据是对象",所以数组可以使用Object类中的to_s方法。此处产生疑问:BasicObject是对象吗?
模块
既然提到父类,那么Class类的父类是什么?
结果为Class类的父类是Module(模块),所以说类就是带有三个方法(new,allocae,superclass)的增强模块。
常量的真相
定义:任何以大写字母开头的引用(包括类名和模块名)都是常量。但Ruby中常量实际上类似变量,尽管Ruby解释器会警告你,但你还是可以修改常量的值。如果你心情不好,甚至可以修改String的类名,从而把整个Ruby系统搞崩溃(interesting~~)。
常量的路径
这两张图可以说的很明白,常量的路径用双冒号进行分隔。其中"::Y"表示路径比较深,双冒号开头表示根位置。
犯晕点:Module类有一个实例方法和类方法,方法名都叫constants。其中Module#constants返回的是当前范围内的所有常量,而Module.constants方法返回当前程序中所有顶层的常量(包括类名)。
对象和类的小结
什么是对象?对象就是一组实例变量外加一个指向其类的引用。对象的方法并不存在于对象本身,而是存在于对象的类中。在类中,这些方法被称为类的实例方法。
什么是类?类就是一个对象(Class类的一个实例),外加一组实例方法和一个对其超类的引用。Class类是Module类的子类,因此一个类也是一个模块。
像其他对象一样,类有自己的方法(比如new方法),这些是Class类的实例方法。像对其他对象一样,类必须通过引用进行访问。