ruby進(jìn)階

  • 相同數(shù)字的object_id不會(huì)變玄呛,相同字符串會(huì)變
1.object_id //200
1.object_id //200

"a".object_id //1200
"a".object_id //1220
  • ruby interpreter(解釋器)將所有的symbol存放在一張symbol table里匿垄,symbol很快,但是不會(huì)被垃圾回收(garbage collecter)
Array.new(3,"abc") // 三個(gè)元素都是一個(gè)引用,改其中一個(gè)其他的都會(huì)變化
Array.new(3) {"abc"} // 不同引用溺忧,改其中一個(gè)不會(huì)影響其他
  • 數(shù)組常用方法
arr = [1,2,3,1,2]
arr.uniq // 去重融欧,[1,2,3]
arr.last / arr[-1]  // 取最后一位
arr.shuffle // 打亂數(shù)組
[[1,2,3],[4,5]].flatten  // [1,2,3,4,5]
arr.each {|value| p value}
arr.each_with_index {|value,index| p "#{index}: #{value}"}
arr.reverse_each {|value| p value}
arr.sort // 排序
arr.select {|value| value>3}
arr.compact // 去掉數(shù)組里的 nil
array.any? {|value| value > 3}  // 數(shù)組里是否存在大于3的值
  • hash 常用方法
h = {a:1 , b:2}
h.assoc :b  // [:b,2]
h.has_value? 2
h.has_key? :b
h.keys // [:a,:b]
h.values  // [1,2]
h.to_a // [[:a,1],[:b,2]]
h.merge({c:3}) // {a:1 ,b:2, c:3}
h.each {|key,value| p [key,value]}
h.each_key {|key| p key}
h.each_value {|value| p value}
h.select {|key| key == :a} // {:a => 1}
h.delete a //刪除a
  • 集合 Set敏弃,無(wú)重復(fù)
Set.new [1,2]
s = _
s.add 'foo'
b = Set.new [2,3]
s & b // {2},求交集
s | b //  求并集
s <= b // 求子集
  • Range
r = 1..2 // [1,2]
r = 1...2 // [1,2)
r.include? 2 //true

a  = [1,2,3,4]
a[1..2] // [2,3]
  • ruby中如何讓 symbol 和 string 實(shí)現(xiàn)相同效果噪馏?麦到??
  • 方法風(fēng)格:在開(kāi)頭把所有會(huì)return的情況都走完
def method x
  return if x.blank? || x.empty?
  func1 ...
  func2 ...
end
  • 單例方法:只對(duì)當(dāng)前對(duì)象有效
str = "hello"
def str.foo
  p self
end
arr = [1,2,3]
def arr.+ num
  self.dup << num
end

arr.+ 4 // [1,2,3,4]
  • 方法別名:讓方法在不同環(huán)境下更加語(yǔ)義化
def foo
  p "foo"
end

alias :bar :foo
  • ruby方法傳參
  1. 默認(rèn)參數(shù)
  2. 任意多參數(shù)
def foo a,*b,c
  p a
  p b 
  p c
end

foo 1,2,3,4,5 
// 1  [2,3,4]  5
  1. hash參數(shù):傳入hash參數(shù)時(shí)欠肾,大括號(hào)可以省略
def foo hash
  h.each do |key,value|
    p [key,value]
  end
end

foo({a:1,b:2})
foo(a:1,b:2) // 傳入hash參數(shù)時(shí)瓶颠,大括號(hào)可以省略
  • load:如果在irb里load一個(gè)ruby文件,不僅可以跑一遍程序刺桃,還會(huì)把文件里的變量或者方法load到當(dāng)前irb環(huán)境里
  • block里的return粹淋,指的是從包含這個(gè)block的方法return,如果這個(gè)block沒(méi)被方法包含瑟慈,是不能return的
  • proc
// & 這個(gè)符號(hào)就相當(dāng)于把block轉(zhuǎn)換為proc桃移,相當(dāng)于一個(gè)方法變量
def foo(&block)
  a = 2
  block.call(a)
end

foo {|v| p v}

---------------------------------

arr = %w(a b c)
arr.map(&:capitalize)  // 這里相當(dāng)于 capitalize這個(gè)method 轉(zhuǎn)為 proc ,proc 再轉(zhuǎn)為 block
  • lambda proc 區(qū)別:
  1. proc參數(shù)可以多傳少傳葛碧,lambda必須傳準(zhǔn)確的參數(shù)個(gè)數(shù)
  2. proc更像block借杰,lambda更像method
  3. return
p = Proc.new {|x| return if x > 0}
p.call(1) // 報(bào)錯(cuò)

l = lambda {|x| return if x > 0}
l.call(1)  // 不會(huì)報(bào)錯(cuò)
  • Class and Object:
  1. Class 可以看做一個(gè) container of methods
  2. Object is a receiver,that can respond to those methods
class Point 
end

p = Point.new

p.methods(false) // 不顯示繼承的方法,只顯示父類的方法
class Point
  attr_accessor :x,:y
  
  def initialize x,y
    @x,@y = x,y
  end

  def first_quadrant?
    x > 0 && y > 0
    // 相當(dāng)于 self.x > 0 && self.y > 0
    // 如果要修改實(shí)例變量的話进泼,必須 @x = 1 或者 self.x = 1
  end

  def +(p2)
    Point.new(x + p2.x, y + p2.y)
  end

  // self在method上就是類方法蔗衡,self在里面就是實(shí)例方法
  def self.second_quradrant?(x,y)
    x < 0 && y > 0
    // 這里如果有self也是指類
  end

  // 批量定義class methods
  class << self
    def foo
    end

    def bar
    end
  end

  @@origin = 0
  // 類變量的getter必須自己定義?
  def self.get_origin
    @@origin
  end

  ORIGIN = 2

end

p1 = Point.new 1,2
p2 = Point.new 2,3

p1 + p2

Point.get_origin
Point::ORIGIN  // 獲取常量的方式
  • ruby中一個(gè)class的所有方法都可以被繼承
  • public protected private
  1. 如果想繼承一個(gè)方法,最好使用public或者protected缘琅,因?yàn)閜rivate也會(huì)被overload粘都,可能造成不期望的后果
  2. 實(shí)例化對(duì)象只能調(diào)用public,protected和private只能在class內(nèi)部調(diào)用
  3. 三者都可以被inheriate
  4. private 只能在內(nèi)部隱式調(diào)用
class A
  def func1
    // 直接調(diào)用不會(huì)報(bào)錯(cuò)
    func1
    func2
  end 

  def func2
  end

  def func3
  end 

  protected :fun2
  private :fun3
end
  1. 在class內(nèi)部刷袍,可以在obj上調(diào)用protected翩隧,不能調(diào)用private
class A
  def func1
    // self就是對(duì)應(yīng)的實(shí)例化對(duì)象,可以call protected,不能call private
    self.func2 // 不會(huì)報(bào)錯(cuò)
    self.fun3  // 報(bào)錯(cuò)
  end

  def func2
  end

  def func3
  end 

  protected :fun2
  private :fun3
end

a = A.new
a.func1
  1. 類方法不能調(diào)用實(shí)例方法堆生,只能調(diào)用類方法专缠,實(shí)例方法中可以訪問(wèn)實(shí)例方法和類方法
  2. 類方法不能被overload,實(shí)例方法(三種)可以
  3. 類方法中不能引用實(shí)例變量淑仆,實(shí)例方法可以引用類變量和實(shí)例變量
  4. 類方法中不能super
  • module只能將instance method mixin進(jìn)來(lái)涝婉,不能mixin class method
  • module可以通過(guò)extend等方法將instance method mixin成class method
module Helper
  def self.a
    p 'a'
  end

  def b
    p 'b'
  end
end

class P
  extend Helper
end

P.b // b
  • 如果既想mixin instance method,又想mixin class method蔗怠,可以使用included這個(gè)hooks:
module Helper
  def a
    p 'a'
  end

  module ClassMethods
    def b
      p 'b'
    end
  end

  // when include in klass, the hook is called
  def self.included(klass)
    klass.extend ClassMethods
  end
end

class P
  include Helper
end

P.included_modules  // [Helper,Kernel]
P.include?(Helper) // true
p = P.new
p.a
P.b
  • ruby可以理解為沒(méi)有class method墩弯,單例方法存儲(chǔ)在對(duì)象的singleton_class中,是該對(duì)象的singleton_classinstance method寞射。因此extend本質(zhì)上就是把method include到class的singleton_class中
str = "hello"
def str.foo
   'foo'
end
str.singleton_class
str.singleton_class.class  // Class
str.singleton_methods // [:foo]
str.singleton_class.instance_methods(false) // [:foo]

class Foo
  def self.foo
      'foo'
  end
end

Foo.singleton_methods // [:foo]
Foo.singleton_class.instance_methods(false) // [:foo]
Foo.singleton_methods == Foo.singleton_class.instance_methods(false) // true
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末渔工,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子桥温,更是在濱河造成了極大的恐慌引矩,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,372評(píng)論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件侵浸,死亡現(xiàn)場(chǎng)離奇詭異旺韭,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)掏觉,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門区端,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人澳腹,你說(shuō)我怎么就攤上這事珊燎。” “怎么了遵湖?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,415評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵悔政,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我延旧,道長(zhǎng)谋国,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,157評(píng)論 1 292
  • 正文 為了忘掉前任迁沫,我火速辦了婚禮芦瘾,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘集畅。我一直安慰自己近弟,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,171評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布挺智。 她就那樣靜靜地躺著祷愉,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上二鳄,一...
    開(kāi)封第一講書(shū)人閱讀 51,125評(píng)論 1 297
  • 那天赴涵,我揣著相機(jī)與錄音,去河邊找鬼订讼。 笑死髓窜,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的欺殿。 我是一名探鬼主播寄纵,決...
    沈念sama閱讀 40,028評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼脖苏!你這毒婦竟也來(lái)了擂啥?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 38,887評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤帆阳,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后屋吨,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體蜒谤,經(jīng)...
    沈念sama閱讀 45,310評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,533評(píng)論 2 332
  • 正文 我和宋清朗相戀三年至扰,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了鳍徽。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,690評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡敢课,死狀恐怖阶祭,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情直秆,我是刑警寧澤濒募,帶...
    沈念sama閱讀 35,411評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站圾结,受9級(jí)特大地震影響瑰剃,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜筝野,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,004評(píng)論 3 325
  • 文/蒙蒙 一晌姚、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧歇竟,春花似錦挥唠、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,659評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春懊烤,著一層夾襖步出監(jiān)牢的瞬間梯醒,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,812評(píng)論 1 268
  • 我被黑心中介騙來(lái)泰國(guó)打工腌紧, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留茸习,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,693評(píng)論 2 368
  • 正文 我出身青樓壁肋,卻偏偏與公主長(zhǎng)得像号胚,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子浸遗,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,577評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容