开发者社区> 贺满> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

Ruby学习_self

简介: self上下文      Ruby的self有和Java的this相似之处,但又大不相同。Java的方法都是在实例方法中引用,所以this一般都是指向当前对象的。而Ruby的代码逐行执行,所以在不同的上下文(context)self就有了不同的含义,先来看看常见的context self都代表哪些...
+关注继续查看

self上下文

     Ruby的self有和Java的this相似之处,但又大不相同。Java的方法都是在实例方法中引用,所以this一般都是指向当前对象的。而Ruby的代码逐行执行,所以在不同的上下文(context)self就有了不同的含义,先来看看常见的context self都代表哪些

# 这个位置位于top level context,代表Object的默认对象main
p self # => main
p self.class # => Object
@self1 = self

# 因为所有自定义类都位于main context之中,所以这是Object的实例方法
# 同时也可以说是一个全局方法
def a_method
  @self2 = self
  p self
  # => main,因为有了实例变量@self1和@self2,所以打印出来的不是main这个字符
  # => 但仍然是main对象,注释掉4,8行即可看到效果
  p @self1 == @self2 # => true
end


# 下面是一个关于类中不同上下文的self
class Person
  p self # => Person,代表当前类
 
  def instance_method
    p self # => #<Person:0xb7818fdc>,代表当前类的实例
  end
 
  def self.class_method
    p self # => Person,和第16行一样代表当前类(这是类方法的context),它们是相等的
  end
end

m = Person.new
def m.hello
  p self # => 代表m这个单例对象
end

m.hello

 

    上面只写了在类中的self,其实在module也是一样的。通过上面代码你可以发现,self一直引用着它所在位置上下文的实例/类。

    self显式/隐式

    你可以先试着运行下面代码,看看有什么意外发生没有


 class Person
  attr_accessor :name
 
  def set_name(your_name)
    name = your_name
  end
end

m = Person.new
p m.name
m.set_name('today')
p m.name # => 猜是什么

 

    如果你猜是today就大错特错了,答案是nil,为什么是nil呢,在第5行,我明明调用的是attr_accessor生成的name=方法赋值的啊,你可以在前面加上self试试,代码如你预期的一样执行了。在这种情况下name = your_name并没有去调用attr_accessor生成的xx=方法,而是将name当作了一个局部变量,如果显式的指定self,就没有问题了。

    读到这,你是不是认为以后这种情况就一直用显式self去调用就好了,其实不然,下面的代码仍会说明一些问题


 class Person

  public
  def get_my_secret1
    my_secret # => 隐式
  end
 
  def get_my_secret2
    self.my_secret # => 显式
  end

  private
  def my_secret
    p 'something...'
  end
 
  def self.secret
    p 'nothing'
  end
end

m = Person.new
#m.my_secret # => private method error
Person.secret # => nothing
m.get_my_secret1 # => something
m.get_my_secret2 # => private method error
 

    上面代码说明:

    第一个问题,显式self不可以调用private(protected的也一样)方法,而隐式的可以,这说明隐式更具有灵活性,但由于上面隐式调用也同时存在一定的“误解”,所以读者应该明白他们之间的区别,以便更好的使用。

    第二个问题,self的方法不受private限制,其实我在这故意误导了读者,因为self的方法是类方法,而权限修饰符只对实例方法生效,所以private的类方法也可以直接访问。

    self“怪异”写法

    下面代码被我个人称为怪异写法,因为平时用不到,但偶尔会看到,但看起来又不太直观,这里列举一下

 class Person
  def metaclass
    class << self
      self
    end
  end
 
  def metaclass2
    self
  end
end

a = Person.new

b = a.metaclass
c = a.metaclass2

# 首先要明白,类Person是Class的一个“实例”,a是Person的一个实例
# 这里b也是一个Person类,但它是独一无二的,即你修改Person不会影响到b,反之亦然
p b # => #<Class:#<Person:0xb76f3800>>
p b.class # => Class

class Person
  def hello
    p 'hello Person'
  end
end

class << b
  def hello
    p 'hello b'
  end
end

b.hello # => hello b


p c # => #<Person:0xb76f3800>
p c.class # => Person

c.hello # => hello Person
 

    还有一个

 class Person
  def self.hello
    p 'hello'
  end
 
  class << self
    # 看了最上面self和context的关系,你应该知道这个self代表是Person类
    # 在这里为Person添加方法,其实也就是为Person添加类方法,和上面的self.hello异曲同工
    def work
      p 'hard work'
    end
  end
end

Person.work

 

 

转自:http://ilstar.blogbus.com/logs/59782933.html
 

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
别梦依稀咒逝川,Ruby二十八年前|M1芯片Mac os系统配置Ruby(3.0.0) on Rails(6.1.1)开发环境
在每个开发者心里,都会有一门“最好”的语言,在这个世界的某个深处,在一些矫矫不群的人们心中,这门语言的名字叫做Ruby,它今年二十八岁了,历史和Java一样的悠久,但是它没有大厂背书、它的性能被开发者诟病、时至今日依然无法高效利用多核资源,甚至于它每年都要被“死亡”一次,相比于有太阳计算机系统、甲骨文、IBM 这些大公司支持的 Java,它是那么的一无所有,但是,它又拥有全世界最虔诚的“信徒”,拥有最活跃的开发者社区,这一切,又让它是那么的应有尽有。是的,这就是Rubyist的理念:有的时候,你想证明给一万个人看,到后来,你发现只得到了一个明白的人,那就够了。
0 0
基于Docker在Win10平台搭建Ruby on Rails 6.0框架开发环境
2023年,“非著名Web框架”--Ruby on Rails已经18岁了。在今年,Rails 6.0趋于完善,除了拿掉讨厌的Jquery,Webpacker 也成为默认前端打包方案,Sprockets 开始软着陆,未来很可能会和Jquery一样被彻底废弃,这就是历史的进程。
0 0
CVE-2018-3760 Ruby On Rails 路径穿越漏洞
CVE-2018-3760 Ruby On Rails 路径穿越漏洞
0 0
CVE-2019-5418 Ruby on Rails 路径穿越与任意文件读取漏洞
CVE-2019-5418 Ruby on Rails 路径穿越与任意文件读取漏洞
0 0
【Ruby on Rails全栈课程】5.5 项目上线--nginx+unicorn部署项目、域名映射
1、域名映射准备材料 云服务器以及域名(参考5.1 购买服务器、域名) nginx unicorn 2、安装以及设置nginx
0 0
【Ruby on Rails全栈课程】5.4 项目上线--项目初始化
1、数据库初始化 (1)登录数据库,创建名为data_system的database
0 0
【Ruby on Rails全栈课程】5.3 项目上线--git部署项目
1、git分布式版本控制系统是什么? 现在本地电脑1有一套data_system的代码,如果想把这套代码放到云服务器上,应该怎么做呢? 最简单的做法是直接将本地电脑1上的代码拷贝到服务器上。 但是我们拷贝完成后发现了一个bug1,在本地电脑1的代码中修复了之后。需要与云服务器上的代码同步,还是要将本地电脑1的代码拷贝到服务器上吗? 团队的另一个人修改了bug2,没有修改bug1,他如果将自己本地电脑2的代码拷贝到服务器上,就会将你刚刚修改的bug1代码覆盖掉。 所以我们现在需要一个第三方先将代码处理整合一下,将我们自己的代码与团队其他人的代码整合起来,再通过git pull操作将整合后
0 0
+关注
贺满
1、7年+测试经险,其中5年+金融(银行)领域业务经验,丰富的app\web\服务端测试经验. 2、4+年管理经验,有多次从0组建测试团队的经历,最多带过20+人的测试团队。
文章
问答
文章排行榜
最热
最新
相关电子书
更多
Python系列直播第一讲——Python中的一切皆对象
立即下载
Python 脚本速查手册
立即下载
PPython高性能计算库--Numba
立即下载