详解Ruby中的单件方法和单件类

所属分类: 脚本专栏 / ruby专题 阅读数: 591
收藏 0 赞 0 分享

单件方法

Ruby允许给单个对象增加方法,这种只针对单个对象生效的方法,称为单件方法

示例代码

str = “just a regular string”

def str.title?
  self.upcase == self
end

str.title? # => false
str.methods.grep(/title?/) # => [:title?]
str.singleton_methods  #=> [:title?]

str.class # => String
String.title? #=> NoMethodError

另外,除了上面使用的定义方法,还可以通过Object#define_singleton_method方法来定义单件方法

单件方法与类方法

前面的笔记中有说道在Ruby中类也是对象,而类名只是常量,所以,在类上调用方法其实跟在对象上调用方法一样:

类方法的实质是: 它是一个类的单件方法,实际上,如果比较单件方法的定义和类方法的定义,会发现其实二者是一样的
 

def obj.a_singleton_method; end
def MyClass.another_class_method; end

二者均使用了def关键词做定义

def object.method
  #方法主体
end

上面的object可以是*对象的引用、常量类名或者self。

类宏attr_accessor

Ruby对象没有属性,如果希望得到一些像属性的东西,需要分别定义一个读方法和写方法(也就是java、objc中的set和get方法),最直接的可以这样:

示例代码

class MyClass
  def my_attribute=(value)
    @my_attribute =value  
  end
  def my_attribute
    @my_attribute
  end
end

obj = MyClass.new
obj.my_attribute = ‘x'
obj.my_attribute  #=> ‘x'

但是上面这种写法,如果属性众多的话就会存在Repeat Yourself的地方,这时就可以用到下面三个类宏:

  • Module#attr_reader 生成一个读方法
  • Module#attr_writer 生成一个写方法
  • Module#attr_accessor 同时生成读方法和写方法

示例代码

class MyClass
  attr_accessor :my_attribue
end

这样是不是就简洁多了呢? 当然,使用方法(读与写)跟上面的实现是一致的。

单件类

我们知道Ruby中对象的方法的查找顺序是: 先向右,再向上,其含义就是先向右找到对象的类,先在类的实例方法中尝试查找,如果没有找到,再继续顺着祖先链找。

前面一篇中有介绍过单件方法,单件方法是指那些只针对某个对象有效的方法,那么如果为一个对象定义了单件方法,那么这个单件方法的查找顺序又应该是怎样的?

class MyClass
  def my_method; end
end

obj = MyClass.new

def obj.my_singleton_method; end

首先,单件方法不会在obj中,因为obj不是一个类,其次它也不在MyClass中,那样的话所有的MyClass都应该能共享调用这个方法,也就构不成单件类了。同理,单件方法也不能在祖先链的某个位置(类似superclass: Object)中。正确的位置是在单件类中,这个类其实就是我们在irb中向对象询问它的类时(obj.class)得到的那个类,不同的是这类与普通的类还是有稍稍不同的。也可以称其为元类或本征类。

打开单件类

Ruby提供了两种方法获取单件类的引用,一种是通过传统的关键词class配合特殊的语法

法一
 

class << an_object
  # 自己的代码
end

obj = Object.new
singleton_class = class << obj
  self
end
singleton_class.class # => Class

另一个方法是,通过Object#singleton_class方法来获得单件类的引用:

法二

“abc”.singleton_class  # => #<Class: #<String:0xxxxxx>>

 
单件类的特性

  • 每个单件类只有一个实例(被称为单件类的原因),而且不能被继承
  • 单件类是一个对象的单件方法的存活所在
  • 引入单件类后的方法查找

基于上面对单件类的基本认识,引入单件类后,Ruby的方法查找方式就不应该是先从其类(普通类)开始,而是应该先从对象的单件类中开始查找,如果在单件类中没有找到想要的方法,它才会开始沿着类(普通类)开始,再到祖先链上去找。这样从单件类之后开始,一切又回到了我们在没有引入单件类时候的次序。

通过下面这个代码可以自行验证一下

class C
  def a_method
    ‘C#a_method()'
  end
end

class D < C; end

obj = D.new

打开单件类定义单件方法

class << obj
  def a_singleton_method
    ‘obj#a_singleton_method()'
  end
end

obj.singleton_class.superclass #=> D

更多精彩内容其他人还在看

Ruby元编程小结

这篇文章主要介绍了Ruby元编程小结,元编程是可以在运行时动态的操作语言结构(如类、模块、实例变量等)的技术,需要的朋友可以参考下
收藏 0 赞 0 分享

Ruby中创建字符串的一些技巧小结

这篇文章主要介绍了Ruby中创建字符串的一些技巧小结,本文用先讲解技巧然后给出代码示例的方式列出了多种Ruby创建字符串方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Ruby Gems更换淘宝源方法

这篇文章主要介绍了Ruby Gems更换淘宝源方法,官方源有时不稳定,国内淘宝做了一个镜像,本文讲解更换成淘宝源的方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Ruby中实现统计文件行数、单词数和字符数

这篇文章主要介绍了Ruby中实现统计文件行数、单词数和字符数,本文是自定义的一个函数,需要的朋友可以参考下
收藏 0 赞 0 分享

Ruby迭代器的7种技巧分享

这篇文章主要介绍了Ruby迭代器的7种技巧分享,Ruby中的迭代器非常人性化,本文既是讲解了7个技巧也是讲解了7种迭代器,需要的朋友可以参考下
收藏 0 赞 0 分享

Ruby和Shell脚本实现判断成绩及格功能

这篇文章主要介绍了Ruby和Shell脚本实现判断成绩及格功能,使用Ruby实现这个功能非常简洁优雅,而Shell的实现就比较传统了,需要的朋友可以参考下
收藏 0 赞 0 分享

Ruby、PHP、Shell实现求50以内的素数

这篇文章主要介绍了Ruby、PHP、Shell实现求50以内的素数,3种语言的实现方法中Shell最简单,PHP最麻烦,Ruby最简洁,需要的朋友可以参考下
收藏 0 赞 0 分享

Ruby简洁学习笔记(一):字符串、数字、类和对象

这篇文章主要介绍了Ruby简洁学习笔记(一):字符串、数字、类和对象,本文是学习笔记第一篇,需要的朋友可以参考下
收藏 0 赞 0 分享

Ruby简洁学习笔记(二):类继承、属性、类变量

这篇文章主要介绍了Ruby简洁学习笔记(二):类继承、属性、类变量,本文是个人学习总结第二篇,需要的朋友可以参考下
收藏 0 赞 0 分享

Ruby中使用连续体Continuation实现生成器

这篇文章主要介绍了Ruby中使用连续体Continuation实现生成器,本文先是介绍了生成器的概念,然后给出实现代码,需要的朋友可以参考下
收藏 0 赞 0 分享
查看更多