1.Symbol类
2.sort方法
3.reduce方法
4.ARGV和gets方法的区别
5.defined?用法
6.&:first用法
7.四种相等
8.unless用法
9.super用法
10.“!”
11.删除方法
12.“::”
13.IO实例
14.异常处理
15.require方法
16.respond_to?和respond_to_missing?
17.define_method
18.public方法的调用
19.send方法调用
20.模块和类的区别
21.计算行数中的闭包问题
22.表达式的求值顺序
23.case用法
24.map和each的返回值
25.赋值和拷贝
1.Symbol类
1.两种形式:一种是直接冒号加名字,一种是冒号加字符串
2.在一文本中,同一Symbol对象表示的类名,方法名,变量指向同一个Symbol对象
3.对于可变的使用String,不可变的使用Symbol
1.两种形式:一种是直接冒号加名字,一种是冒号加字符串
#下面两种就是Symbol对象的两种表现形式
:test
:"test"
#上面的两种形式表示的是同一个对象,就是具有相同的object_id,指向同一个内存地址。
:test.object_id #=>355868
:"test".object_id #=>355868
#即是名字相同,symbol相同,:test和:“test”是形同的名字,所以他们是相同的symbol。
#但是如果是:@test和:test是不同的名字,所以他们是不同的symbol。
:@test.object_id # 1148868
:test.object_id # 355868
#区别于两个值相同的字符串,他们只是值相同,但不是同一个字符串对象,没有指向同一个内存地址
“test”.object_id #=> 70114101202320
"test".object_id #=> 70114107991560
2.在一文本中,同一Symbol对象表示的类名,方法名,变量指向同一个Symbol对象
module One
class Fred
end
$f1 = :Fred
end
module Two
Fred = 1
$f2 = :Fred
end
def Fred()
end
$f3 = :Fred
$f1.object_id #=> 2514190
$f2.object_id #=> 2514190
$f3.object_id #=> 2514190
3.对于可变的使用String,不可变的使用Symbol
String有"[]="方法,Symbol没有这个方法,因为它是不可变的。
使用Symbol方法可以降低内存,因为相同名字的Symbol指向的是同一个对象。
2.sort方法
a = [1,2,3]
a.sort #=>[1,2,3]
a.sort{|x,y|x<=>y} #=>[1,2,3]
a.sort{|x,y|y<=>x} #=>[3,2,1]
#前值和后值做比较(就是x<=>y的形式),如果前者比后者小,则按照[x,y]进行排序,就是所谓升序排列。
#如果是后值和前值做比较(就是y<=>x的形式),就是所谓的降序排列。
3.reduce方法
作用:遍历数组中的每个值进行累加,累加值即是最后的结果。
#只添加symbol
(5..10).reduce(:+) #=>45 遍历每个数进行累加,返回最终的累计值
#添加初始值和symbol
(5..10).reduce(1,:+) #=>46 1是初始值
#添加block,在没有初始值的情况下,数组的第一个值初始值,即sum=5,从第二个值开始累加
#从第二个值进行累加,这是自己的理解
(5..10).inject { |sum, n| sum + n } #=>45
#添加block和初始值,从初始值开始累加
(5..10).inject(1) { |sum, n| sum + n } #=>46
4.ARGV和gets方法的区别
1.ARGV的用法:
#demo.rb:
puts "首个参数:#{ARGV[0]}"
#在终端执行代码
demo.rb demo
#结果返回
首个参数:demo
2.gets的用法:
#demo.rb:
puts "首个参数:#{gets()}"
#执行
demo.rb demo #无效
#执行
demo.rb #回车
demo # 首个参数:demo
总结:ARGV和gets这两个方法主要的区别是“回车”
5.defined?用法
defined?和super一样,都是关键词,不是方法
1.defined? 是一个特殊的运算符,以方法调用的形式来判断传递的表达式是否已定义。
2.它返回表达式的描述字符串,如果表达式未定义则返回 nil。
#用法一:判断变量是否定义
foo = 42
defined? foo # => "local-variable"
defined? $_ # => "global-variable"
defined? bar # => nil(未定义)
#用法二:判断方法是否定义
defined? puts # => "method"
defined? puts(bar) # => nil(在这里 bar 未定义)
defined? unpack # => nil(在这里未定义)
#用法三:判断yield是否可用
若 yield 调用可用,则返回真,具体返回值为字符串 "yield"
class Base
def foo
puts defined? yield
end
end
a = Base.new
a.foo # =>nil
a.foo{ this is the block} # => yield
#用法四:判断super是否可用
defined? super,若 super 可被调用,则返回真,具体返回值为字符串 "super"
class Base
def foo
end
end
class Derived < Base
def foo
puts defined? super
end
def fun
puts defined? super
end
end
obj = Derived.new
obj.foo #=> super
obj.fun #=> nil
6.&:first用法
array = [["a",5], ["b", 3], ["a",6]]
array.uniq(&:first).sort_by(&:last)
#上面的代码等同于
array.uniq{|a|.a.first}.sort_by{|x|x.last}
#就是按照数据的第一个元素去重,然后按照数组的第二个元素排序
7.四种相等
1."=="表示值形同
2."===",一般在case中,表示某种匹配
3."equal?"是指两个对象的object_id值一样,即是引用同一个对象。
4."eql?"要求值和对象类型都必须相同。
1. "=="表示值形同
#字符串的内容相同
a = "string"
b = "string" + ""
a == b #=>true
#数值相同
a = 1
b = 1.0
a == b #=>true
2. "===",一般在case中,表示某种匹配
case some_object
when /a regex/
# The regex matches
when 2..4
# some_object is in the range 2..4
when lambda {|x| some_crazy_custom_predicate }
# the lambda returned trueend
#上面的代码和下面代码的实现形式:
if /a regex/ === some_object
# The regex matches
elsif (2..4) === some_object
# some_object is in the range 2..4
elsif lambda {|x| some_crazy_custom_predicate } === some_object
# the lambda returned trueend
3. "equal?"是指两个对象的object_id值一样,即是引用同一个对象。
obj = "a"
other = obj.dup
obj == other #=> true
obj.equal? other #=> false
obj.equal? obj #=> true
a = "string"
b = "string" + ""
a.object_id #=>70268974328100
b.object_id #=>70268970910720
a == b #=>true #a和b的数值相同
a.equal?b #=>false #a和b有不同的object_id
a = 1
b = 1.0
a == b #=>true #=>a和b的数值相同
a.object_id #=>3
b.object_id #=>-36028797018963966
a.equal?b #=>false #a和b具有不同的object_id
4. "eql?"要求值和对象类型都必须相同
a = "string"
b = "string"+""
a.eql?b #=>true
#上面的a和b两个值相同,并且属于String类
a = 1
b = 1.0
a.eql?b #=>false
#上面的a和b两个值相同,但是1属于Integer类,b属于Float类
8.unless用法
unless相当于if not ,如果unless后面的语句执行的结果为false,则该语句执行,如果为true,则该语句不执行。
#代码如下所示,当unless后面为false的时:
def demo
puts "test" unless false
end
demo #=>这个执行结果为test
#当unless后面为true时:
def demo
puts "test" unless true
end
demo #=>这个执行结果为nil
9.super用法
super是调用父类方法的执行,但是super和super()的调用结果不同。
1.super()表示不带参数,不把参数带入到父类中。
2.super表示需要将参数带入到父类中,并且执行相应的方法。
3.使用super时,从继承体系中的上一层寻找和当前同名的方法,同样适用于module
4.报错之后,super之后的代码不执行
5.使用super时引发method_missing方法
1.第一种:使用super()方法:
class Foo
def show
puts "Foo#show"
end
end
class Bar < Foo
def show(text)
super()
puts text
end
end
Bar.new.show("test")
#下面是结果
Foo#show
test
#使用super()方法,不将参数带入到父类的show方法中,父类的show方法也不存在调用参数。
2.第二种:使用super方法:
class Foo
def show
puts "Foo#show"
end
end
class Bar < Foo
def show(text)
super
puts text
end
end
Bar.new.show("test")
#wrong number of arguments (given 1, expected 0) (ArgumentError)
上面的代码中super方法会将text这个参数传递给父类的show方法,但是父类的方法其实没有参数选项,因此出错。
#代码修正
def show(text)
puts "#{text}, Foo#show"
end
3.同样适用于module
module One
def demo
puts "this is the method one demo"
end
end
module Two
include One
def demo
super
puts "this is the method two demo"
end
end
class A
include Two
end
obj = A.new
obj.demo
#结果
this is the method one demo
this is the method two demo
4.使用super报错之后不会执行super之后的语句
class Demo
def method_missing(name, *args, &block)
super
puts "this is the method_missing"
end
end
obj = Demo.new
obj.cc #undefied method cc
#并且super之后的代码不执行
5.使用super时引发method_missing方法
class Father
def laugh
super
end
end
Father.new.laugh #=>no superclass method `laugh' NoMethodError
#如果在继承体系中定义了method_missing方法,将丢失默认的提示信息
class Father
def laugh
super
end
def method_missing(method)
puts "this is the missing method"
end
end
Father.new.laugh #=> this is the missing method
10.“!”
compact
#返回了去除了nil的对自身拷贝的数组,但是自身数组其实没有变化
Returns a copy of self with all nil elements removed.
a = ["a","b",nil].compact
a.compact #=>["a", "b"]
a #=>["a","b",nil]
compact!
#改变了原来的数组
Removes nil elements from the array.
Returns nil if no changes were made, otherwise returns the array.
#如果存在nil,则返回去除了nil的数组
a = ["a","b",nil]
a.compact! #=>["a","b"]
a #=>["a","b"]
#如果不存在nil,则返回nil
a= ["a", "b"].compact! #=>["a", "b"]
a #=>["a","b"]
11.删除方法
两个删除方法,包括undef_method和remove_method方法
1.前者方法会删除所有包括继承来的方法,而后者只删除接受这自己的方法,保留继承的方法
2.这两个方法都是Module类的实例方法,可以在类中直接使用。
class Parent
def demo
puts "this is the parent"
end
end
class Child
def demo
puts "this is the child"
end
end
obj = Child.new
obj.demo #=> "this is the child"
#使用remove_method方法
class Child
remove_method :demo
end
obj =Child.new
#说明删除了自己的方法,保留了继承的方法
obj.demo #=>"this is the parent"
#使用undef_method方法
class Child
undef_method :demo
end
obj =Child.new
#说明删除了所有的方法,包括继承来的方法
obj.demo #=>undefined method demo
12."::"
示例:
module Foo
class Bar
A = “demo one”
end
B = "demo two"
class Tao
delf self.method2
"method2"
end
end
class Zen
def method2
"method2"
end
end
class ::FooBar
def self.method1
"method1"
end
end
end
puts Foo::Bar::A #调用类名
puts Foo::B #调用常量
puts Foo::Tao::method1 #调用类方法
puts Foo::Zen::new::method2 #调用实例方法
FooBar::method1 #::表示 root namespace
#第一种情况:双引号是定义namespace或者叫做scope用的,当使用Foo::Bar的时候,实际上是在找名字叫做Foo的namespace,然后返回里面的Bar参数
这个Bar可以是个常量,可以是个类,可以是个方法(类名和方法名在Ruby中被视为常量)
#第二种情况:“::”还能用来找真正的常量
class Foo
Bar = "hello" #常量
bar = "hello" #局部变量
end
Foo::Bar #=>"hello"
Foo::bar #=>出错
Foo.Bar #=>出错
Foo.Bar #=>出错
#第三种情况:“::”在开始的位置则表示回到root namespace,不管前面套了几个module,而这个root namespace是Object
puts self #=>main
puts self.class #=> Object
#第四情况:一般情况下是module做为namespace,但是其实class其实也可以作为namespace:
class Externel
#class Internel换成module Internel效果也是一样的
class Internel
def self.class_method
puts "this is the class method"
end
def instance_method
puts "this is the instance method "
end
end
end
Externel::Internel.class_method #=>"this is the class method"
Externel::Internel::new::instance_method #=> "this is the instance method"
13.IO实例
#####实例:写Blog
class Blog
attr_accessor :author, :title, :time, :content
def initialize(author = "blackanger", title, content)
@author = author
@title = title
@content = "#{content}"
@time = Time.now.strftime("%Y-%m-%d %H:%M:%S")
end
def write
File.open("blog_date", 'a+') do |f|
enter(f)
f.write DATA.read
enter(f)
f.write @time
enter(f, 2)
f.write "Title: #{@title}"
enter(f, 2)
f.write "Content:"
enter(f)
f.write @content
enter(f)
end
end
def self.read
File.open("blog_date", "r") do |f|
f.each_line{|el| puts el}
end
end
def enter(f, n=1)
n.times{f.write "\n"}
end
end
if ARGV[0] == "read"
Blog.read
else
puts "opening the blog data file .."
puts "请输入标题:"
title = STDIN.gets.chomp
puts "请输入内容:(输入END结束内容)"
content = ""
STDIN.each_line do |str|
content << str
break if str.chomp == "END"
end
puts "OK,正在写入文件.."
blog = Blog.new(title, content)
blog.write
puts "提交成功! 下面是您的Blog!"
puts "当前目录:#{__dir__}, 当前文件:#{__FILE__}"
Blog.read
end
__END__
#------
#copyright
#------
14.异常处理
#代码示例
begin
1/m
raise "error A"
#fail "Error"
rescue =>e
puts "#{e.message}"
raise "error B"
m +=1
retry
else
puts "test"
ensure
puts "must to do"
end
1. begin .. end 语句可以放在方法里面,也可以放在类里面
#如果放在方法里面可以省略保留字begin,end,直接书写rescue和ensure:
def foo
rescue =>ex
异常处理
ensure
后处理
end
#如果放在类里面,也可以直接使用rescue和ensure,省略begin和end
#但是的话如果类定义途中发生异常,那么异常发生部分后的定义方法就不会再执行,因此一般不在类定义中使用他们
2. 这里几个关键词比如begin,rescue,ensure,retry,raise,fail,else等,其中begin,rescue,ensure,retry,else是保留字,而fail和raise是Kernel的module method,其两者的功能都是一样的,都是抛出相应的内容。
3. ensure的作用是无论上面的代码是否执行,ensure里面的代码都会执行。
4. rescue只接受异常类,而符号=>与hash中的符号的表示意义不一样,这里表示的意思是传递。
5. retry是重新执行这段代码的意思,而这里被重新执行的代码是begin到rescue这段代码,使用retry容易产生死循环。
6. else表示没有rescue的时候进行执行else里面的代码。
7. rescue中的代码如下所示,它一般捕获的异常类。
rescue =>e
#一般是这种形式,如果没有指明是哪个异常类的话,默认是使用StandardError类
#如果按照rescue =>e 这种形式来写的话,如果实际报出的错误不是StandardError类的对象的话,则这个错误捕获不到。
rescue ZeroDivisionError =>e #也可以表示成这种形式
具体代码如下:
def divide(m)
begin
1/m
rescue ZeroDivisionError => e
m += 1
retry
end
end
divide 0 #=>1
#接上上面的代码继续进行说明,如果 =>e前面没有指明类,默认是StandardError类的对象,但是如果捕获LoadError异常的话,代码报错:
def divide(m)
begin
require "xxx"
1/m
rescue =>e
puts "#{e.class}"
end
end
#执行上面的代码会报错,但是执行下面的代码会出现预期的结果
def divide(m)
begin
require "xxx"
1/m
rescue LoadError =>e
puts "#{e.class}"
end
end
8.rescue可以捕获多个异常,见如下的代码所示:
def divide(m)
begin
require "xxx"
1/m
rescue LoadError => e
puts "#{e.class}"
rescue ZeroDivisionError => e
puts "#{e.class}"
end
end
divide(0) # LoadError
9. rescue的另外一种用法
def divide(m)
1/m rescue $!
end
divide(0) # ZeroDivisionError: divided by 0
10. ensure不会影响begin的返回值:
def divide(m)
begin
1/m
rescue ZeroDivisionError => e
puts "#{e.message}"
m += 1
retry
ensure
puts m*2
end
end
a = divide 0
puts a # =>1,返回值即是begin中的返回值
11. 在ensure中添加return关键词,则影响方法的返回值:
def divide(m)
begin
1/m
rescue ZeroDivisionError => e
puts "#{e.message}"
m += 1
retry
ensure
return m*2
end
end
a = divide 0
puts a # =>2,方法的返回值变为return的值
begin
1/m
rescue =>e
puts "#{e.message}"
m += 1
retry
else
puts "something"
ensure
puts "must to do"
end
#如果方法调用的过程中m取值为0,则会报出异常,报出异常的方式其实和raise方法一致:
raise "error A" if m == 0
代码的具体执行步骤如下:
1.create Exception Object
2.set traceback
3.set global exception variable $!
15.require方法
#require方法
如果require的文件不是绝对路径,那么从$LOAD_PATH($:)中寻找相应的文件
可以在终端中执行$LOAD_PATH,查看其中的内容
#relative_require方法
如果要应用相同目录下的文件内容,需要使用relative_require方法
16.respond_to?和respond_to_missing?
#method_missing不友好,它能够代理方法,但是respond_to?的过程中,结果是false:
class Demo
def method_missing(method, *args, &block)
if method.to_s =~ /play_(\w+)/
puts "here is #{$1}"
else
super
end
end
end
p = Demo.new
p.play_with_girl #=>here is with_girl
p.respond_to? :play_with_girl #=> false
#为了让respond_to?返回true,可以使用下面的语句:
class Demo
def method_missing(method, *args, &block)
if method.to_s =~ /play_(\w+)/
puts "here is #{$1}"
else
super
end
end
def respond_to?(method, *)
method.to_s =~ /play_(\w+)/ || super
end
end
p = Demo.new
p.play_with_girl #=>here is with_girl
p.respond_to? :play_with_girl #=> 0
#但是其实上面的play_with_girl还是不像一个方法,见如下代码:
p.method :play_with_girl #=> NameError,undefined method
#为了解决上面的问题,引入respond_to_missing?方法,下面是完整的解决方案:
class Demo
def method_missing(method, *args, &block)
if method.to_s =~ /play_(\w+)/
puts "here is #{$1}"
else
super
end
end
def respond_to_missing?(method, *)
method =~ /play_(\w+)/ || super
end
end
p = Demo.new
p.play_with_girl #=> here is with_girl
p.respond_to? :play_with_girl #=> true
m = p.method(:play_with_girl)
m.call #=> here is with_girl
m.name #=> play_with_girl
Demo.send :define_method, :hello, m
p.hello #=> here is with_girl
17.define_method
#一般用法:下面的代码直接在类中使用define_method方法定义:test的类
class Test
define_method :test do
puts "this is the define_method demo"
end
end
obj = Test.new
obj.test #=>"this is the define_method demo"
疑惑:为什么可以在类中直接使用define_method方法,以为是实现了module_funciton的功能
module_functon主要实现了下面的两个功能:
1.模块名.方法名直接调用
Kernel.puts "this is the test"
2.类似于关键字在程序的任何部分进行调用
puts "this is the test"
但是,其实在define_method不能实现上面的两个功能
最终解释:
1.define_method是Module类私有实例方法,同时Module类是Class类的父类。
2.define_method做为一般类对象的类方法被直接调用。
#irb: Module.private_instance_method.grep(/define_method/)
#irb: [:define_method]
class Module
def demo
puts "this is the demo"
end
private :demo
end
class Test
#对象类中的私有方法,在对象中可以被直接调用,而且必须是隐式调用
demo
end
18.public方法的调用
1.public的类方法和实例方法可以被显示调用,也可以被隐式调用。
2.这个方法必须在所在的类或者实例化对象的作用域内。
#公开的类方法可以被显示调用,可以被隐式调用:
class DemoClass
class << self
def demo_method
puts "this is the class method"
end
end
DemoClass.demo_method #如期执行
demo_method #如期执行
end
demo_method #会报undefined method or variable
#上面的方法不在DemoClass这个作用域中,而在Object这个类的作用域中。
#公开的实例方法可以别显示调用,也可以被隐式调用:
class DemoClass
def demo_method
puts "this is the instance method"
end
end
obj = DemoClass.new
obj.demo_method
obj.instance_eval{ demo_method }
19.send方法调用
send是BasicObject类的public instance method,可以被调用的对象包括:
1.普通对象
2.普通类
3.普通模块
4.Kernel模块
class DemoClass
class << self
def class_method
puts "this is the class method"
end
end
def demo_method
puts "this is the demo"
end
end
obj = DemoClass.new
obj.demo_method
obj.send(:demo_method) #可以被普通对象调用
Kernel.send(:puts, "cc") #可以被Kernel模块调用
DemoClass.send(:class_method) #可以被普通类调用
module Mod
class << self
def module_method
puts "this is the module method"
end
end
end
Mod.send(:module_method) #可以被普通模块进行定义
#send方法可以被普通对象调用
DemoClass.superclass #=> Object
Object.superclass #=>BasicObject
#类DemoClass间接继承于BasicObject类,继承send方,DemoClass的普通对象可以调用BasicObject类的实例方法send。
#send方法可以被普通类进行调用
DemoClass.class #=>Class
Class.superclass #=>Object
#send方法是Class类的实例化方法,亦是DemoClass类的类方法。
#send方法可以被普模块调用
Mod.class #=>Module
Module.superclass #=>Object
#send方法是Module类的实例方法,是其实例化对象即普通模块的模块方法。
#send方法可以被Kernel模块调用
#Kernel模块是Module类的一个普通实例化对象,代码如下所示:
Kernel.class #=>Module
20.模块和类的区别
1.在类中,类的实例方法是其类对象的类方法。
2.因为类和模块相比,只是增加了三个方法(new, allocate,superclass)
#类的形式
class String
def self.inherited(subclass)
puts "#{self} was inherited by #{subclass}"
end
end
class MyClass < String; end #=> String was inherited by MyClass
#inherited是Class类的私有实例方法:
Class.private_instance_methods.grep(/inherited/) #=>[:inherited]
#String是Class的实例化对象,因此Class的实例方法也就成了String类(本质上是一个对象)的类方法。
#模块的形式
module M; end
M.class #=>Module
module M
def self.incluede(othermod)
puts "M was mixed into #{othermod}"
end
class C
include M
end
end
#include方法是类Module的私有实例方法:
Module.private_instance_methods.grep(/include/) #=>[:include]
#普通的模块其实是类module的对象,因此moudle类的实例方法是其实例化对象(也就是一个普通模块)的模块方法
所以,类和模块在原理上其实是一致的。
21.计算行数中的闭包问题
闭包:是引用了一个作用域外的变量并且可以被调用的函数。
ltotal = 0
ARGV.each do |file|
begin
input = File.open(file)
l = 0
input.each_line do |line|
l += 1
#ltotal += l #最终执行结果为153
#检测这个问题的方法是将如下的结果在控制台打印出来
#puts l
#puts ltotal
end
input.close
ltotal += l #最终执行结果为17
rescue =>e
puts "#{e.message}"
end
end
puts ltotal
22.表达式的求值顺序
1.解释器对每个表达式都是从右向左求值的
2.同一个作用域的方法才能调用,不同作用域的方法不能调用
3.send方法接受返回字符串对象和符号对象的方法名的方法也是可以的。
1.解释器对每个表达式都是从右向左求值的
class Computer
def mouse
component :mouse
end
def component(name)
puts "get the #{name} information in the computer"
end
end
cp = Computer.new
cp.mouse
#返回预期的结果:
def mouse
component :mouse
end
#更改代码
def mouse
component(mouse) #出现SystemStackError错误
end
#解释器对每个表达式都是从右向左求值的的
def mouse
component :mouse#将:mouse理解为符号对象
end
#更改代码
def mouse
component(mouse) #将mouse理解为方法,因此出现如上死循环错误
end
#出现如下的错误
component(mouse)
component(component(mouse))
component(component(component(mouse)))
#改动代码
def mouse
component cc #undefined local variable or method "cc"
end
#解释:解释器对表达式是从右向左求值,但是cc这个变量没有被定义过,按照这个形式,有可能是方法,也有可能是局部变量。
2.同一个作用域的方法才能调用,不同作用域的方法不能调用
class MyClass
def my_method(my_arg)
puts my_arg*2
end
end
obj = MyClass.new
obj.send(:my_method, 3) #=>6
obj.send("my_method", 6) #=>6
obj.send(my_method, 6) #=> undefined variable or method error
#调用方法需要考虑作用域的
class Computer
def mouse
component mouse
end
def component(name)
puts "get the #{name} information in the computer"
end
end
#component方法调用mouse方法,这两个方法在同一个作用域中
#调用my_method方法的时候,会在该方法的作用域中寻找该方法,但是my_method方法跳出MyClass之后,就失效了,该方法的作用域是顶级作用域
#区别obj.send(my_method)和obj.my_method
obj.send(my_method, 6) #=> undefined variable or method error
3.send方法接受返回字符串对象和符号对象的方法名
class MyClass
def my_method(my_arg)
puts my_arg*2
end
end
obj = MyClass.new
def simulate
:my_method
end
obj.send(simulate, 6) #=>6
23.case用法
#第一种情况类似于if语句
year=2012
leap = case
when year%400 == 0 then true
when year%100 == 0 then false
else year%4 == 0
end
puts leap
#第二种情况case语句的顶部指定一个目标,而每个when从句列出一个或者多个比较条件
input="run"
case input
when "debug"
print "debug time"
when "run"
print "run now"
else
print "wrong"
end
24.map和each的返回值
需要取出tag数组中的name值,并且作为字符串显示,两个方法都是进行遍历,但是each返回的还是原来的数组,map返回的是遍历后的数组,所有后者能达到所要的效果,两个方法都不能改变数组本身。
#map
a=[[1,"a"], [2,"b"]]
a.each(&:first).join(",") #=>[[1,"a"], [2,"b"]]
b.each(&:first).join(",") #=>"1,2"
25.赋值和拷贝
赋值是创建一个引用,如果将一个对象赋值给另外一个对象,那么这两个对象指向同一个引用:
str1 = 'hello' # => "hello"
str1.object_id # => 17940720
str2 = str1 # => "hello"
str2.object_id # => 17940720
两个对象均指向同一个引用,如图所示:
如果修改其中一个对象,则会改变其引用,也即是改变另外一个对象。但是还是指向的是同一个引用。
str2 << '!' # => "hello!"
str1 # => "hello!"
str2 # => "hello!"
str1.object_id #=>70296829690720
str2.object_id #=>70296829690720
如果对其中一个对象重新复制,则两个对象不再指向同一个引用。
str2 = 'ruby!' # => "ruby!"
str2.object_id # => 17304640
str1.object_id # => 17940720