写这篇文章的初衷源于我的伙伴们在上手Ruby过程中,表现实在是太让人拙计了。由于项目的急功近利,需要迅速入门Ruby并上手项目。所以很多开发者在实际开发过程中,不熟悉Ruby的表达方式,也会沿用其他语言比较生涩的表达形式。在我看来,Ruby的学习应该是个系统的循序渐进的过程,尽量避免急迫的方式。不过现实往往是紧迫的,所以就有了这篇文章。
标题中的菜鸟主要是指那些有其他编程语言开发经验,但在Ruby面前是个学习时间不久的初学者。我的工作伙伴都是这样的一些人。所以,我就在考虑Ruby和其他语言的一些显著的不同点,想要提点他们注意这些地方,能将程序尽量写得Rubic。在我与其他语言进行比较的过程中,有了一些灵感,所以我就写了点。后来灵感多了些,我也就多写了些。
着重指出一点,没有菜鸟开发者,只有停滞不前的开发者。系统学习一门语言才能自如地运用该语言的表达。Ruby的官方文档是永远是学习Ruby语言的最好资料,它最全面且持续更新。此外,Ruby的官方也给出了从其它语言到 Ruby的比较文档。
字符串
在Java中,字符串只有双引号的形式;在Python和JavaScript中,字符串既有单引号形式也有双引号形式,但它们实践是一回事。在Ruby的世界中,字符串的表达方式要丰富得多。Ruby中的字符串,既有单引号的形式,也有双引号的形式,但它们的含义却是截然不同的。
单引号的字符串,指的是原生字符串,其中不含有转义规则。例如'\n'
不是指换行符,而是指两个字符的字符串,分别为字符\\
和字符n
。
双引号的字符串,是包含转义规则的。例如"\n"
就是指换行符。另外,双引号的字符串还可以使用#{...}
的方式嵌入表达式。
"My Name is #{name}, and my years old is #{age}."
TODO 多行字符串
符号
Ruby引入符号的概念,是源于Ruby的一个设计缺陷。在大部分语言设计中,字符串都设计为不可变对象;然而在Ruby当中,字符串被设计为可变的。为此,引入了另一个字符序列的类型,但它是不可变的。这就是符号(Symbol)。
Ruby中符号的写法是在文本前面加上冒号(:)。对于Ruby中合法的变量名,直接加上冒号即可,例如:
:name
:age
:kind_of_animal
也可以在字符串前面加上冒号,这时符号的内容可以包含空格等变量名不允许的字符。示例:
:"My Name is #{name}, and my years old is #{age}."
这里字符串的单引号和双引号的性质也被沿用了下来。
符号是不可变的;并且,其是唯一的。对于内容完全相同的符号,其只会有一个实例存在于Ruby解释器环境中。
TODO 相同内容符号的object_id一致
符号对于Hash的构造有特殊的作用。Ruby中,Hash的构造比较繁琐,要用到=>
符号:
{ 'a' => 'a', 'b' => 'b', 'c' => 'c' }
但是如果键值都是符号的话,有简便的构造方法:
{ a: 'a', b: 'b', c: 'c' }
它等价于:
{ :a => 'a', :b => 'b', :c => 'c' }
由于在方法调用中最后一个Hash参数可以省略大括号,所以下面的调用
f({ a: 'a', b: 'b', c: 'c' })
可以去掉大括号,写成:
f(a: 'a', b: 'b', c: 'c')
这就很像Python的关键字参数了。
最后,符号和字符串的相互转化方式如下:
str.to_sym #字符串转化为符号
sym.to_s #符号转化为字符串
方法名是符号。
一切皆是对象
Java中有些基本数据类型(boolean、int、double),它们不是对象。JavaScript中的基本数据类型可以理解为对象,如boolean、number、string、function;但它们不能像普通的JS对象那样自如地增删属性。Python要彻底些,一切皆是对象,包括数字和布尔值都是普通的对象。
不过,比起所有其他的语言,Ruby是贯彻一切皆是对象这一思想最彻底的语言。在Ruby中:
- 数字、布尔值、字符串、数组是对象。
- 类是对象,它是Class类的实例。
- 指示对象不存在的nil值也是对象,它是NilClass的实例,可以调用方法to_s, to_i等。
一切皆是表达式
这可能是Ruby又一个玄妙的地方了,影射了Ruby大一统的设计理念。Ruby中一切皆是表达式,一切表达方式都返回值。
- 普通的Ruby表达式当然返回值。
- if、while等控制结构返回值,其值是内部最后一条语句的值。
- 方法调用返回值,其值由return指定或者是方法内最后一条语句的值。
- 类定义返回值,其值是类内部最后一条语句的值。
- 方法定义返回值,其值是方法名表示的符号。
TODO 一切皆是表达式示例
一切皆是方法调用
Ruby是也是贯彻面向对象设计模式最彻底的语言。在面向对象的设计理念中,为了提升封装性,会禁止直接的对象属性操作,只给出相应的get和set方法。
而在Ruby中,从语言层级中就禁止了直接访问对象属性的途径,只存在方法调用。我们平常常见的a.name
和a.name = 'Hello'
的调用方式,看上去很像是直接操作属性,其实不是的。因为Ruby的方法调用可以省略括号,所以
a.name
等价于
a.name()
而
a.name = 'Hello'
的形式,实际上是
a.name=('Hello')
的简写。
TODO attr_reader, attr_writer, attr_accessor宏
省略分号
TODO 省略分号
省略括号的方法调用
Ruby中的方法调用,可以省略括号:
f a, b, c
等价于:
f(a, b, c)
控制结构
TODO 控制结构:if-else,unless-else,when-case,while,begin-rescue-ensure
变量命名约定
面向对象语言中,根据作用域的不同,分别有全局变量、常量、类变量、实例变量、局部变量的区别。在其他的语言中,可能是通过不同的修饰符和作用域进行区别。例如在Java中,没有全局变量的概念,常量用final修饰,类变量用static修饰,实例变量存在于类定义中,局部变量存在于方法定义中。有时候,这些变量很容易混淆;局部变量也很容易覆盖掉实例变量和类变量。
在Ruby中,是以变量名的格式区别变量的类型。总结如下:
- 局部变量:以小写字母开头
- 常量:以大写字母开头
- 实例变量:以@开头
- 类变量:以@@开头
- 全局变量:以$开头
TODO 变量名示例
方法名约定
Ruby中的方法名结尾可以跟?和!。一般来说,以?结尾的方法返回布尔值;以!结尾的方法提示危险性。例如字符串中,empty?
返回字符串是否为空;sub
和sub!
方法替换字符串,区别是sub
不修改原字符串,返回替换的字符串;sub!
作原地替换,会修改原字符串。
一般来讲,返回布尔值的方法名以?结尾,会对原对象作潜在性的修改的方法名以!结尾。
块
Ruby中的方法不是一等公民。方法不是值,不能作为参数传递给另一个方法。不过Ruby有块的概念,块可以作为语言特性附加给方法调用。
块的写法有两种形式,分别是大括号形式和do...end形式。
大括号形式:
{ |a, b|
a.to_i <= b.to_i
}
do...end形式:
do |a, b|
a.to_i <= b.to_i
end
乍一看,大括号的形式有点丑。一般来说,只在单行块的时候用大括号的形式:
{ |a, b| a.to_i <= b.to_i }
Ruby中的每个方法都可以接受一个块。块要紧接着方法调用的后面,不是方法的参数之一。例如
f(1, 2, 3) do |a, b|
a.to_i <= b.to_i
end
上面的示例中省略括号也是合法的:
f 1, 2, 3 do |a, b|
a.to_i <= b.to_i
end
但像下面这样把块用括号括起来作为最后一个参数就是错误的:
f(1, 2, 3, { |a, b| a.to_i <= b.to_i }) #语法错误
块与方法一个很大的区别是块支持闭包:
min = 18
array.select do |v|
v >= min #访问外层min变量
end
TODO 在方法定义中用yield关键字调用块
迭代器
Ruby中没有提供Java和JavaScript当中三段for循环的模式。所谓三段for循环,是指下面JavaScript的for循环:
for(i = 0; i <= 9; i++) {
console.log(i);
}
Ruby中没有提供这样的方式。但这并没有什么痛苦的,因为Ruby提供了基于块的迭代方式。例如上面的代码,在Ruby中可以写成:
0.upto(9) do |i|
puts i
end
Ruby中的集合,都支持这样传递代码块的迭代方式。Ruby中的集合包括Range、Array、Hash、String、etc。例如上面的示例可以改写成Range的each迭代形式:
(0..9).each do |i|
puts i
end
TODO -ect方法:collect, inject, select
真假值
Ruby中用于if、unless、while的条件判断中,真假值的规则很简洁:
- nil和false为假
- 其余都为真
TODO list
我懒!