写在前面
我们当然希望能在本地中执行代码,如果你还没有只是看看,还没安装 ruby,可以在 try ruby 或者 replit 在线编写代码
本文目录
Ruby 是什么
基本语法
变量
算术操作
位运算符
打印输出
注释
条件语句:控制流
循环/迭代器
数据结构
String
Integer/Float
Array
Hash
Set
Range
函数(方法)
面向对象编程
对象
类
Ruby 是什么
弱类型语言
Ruby 是一门弱类型语言,和 JavaScript 一样
a = 5;a = 'Hello'puts a # Hello
命名风格
像变量(variable)、符号(symbol)、方法(method),通常使用 snake_case 风格
snake_case 风格即短语内的各个单词之间以下划线做间隔
像常数(constant),使用 CONST_FOO 风格
类名(class name),使用骆驼 (CamelCase)风格
文件名(file name),使用 snake_case 风格
“$”开头的变量:全局变量
“@”开头的变量:实例变量
“@@”开头的变量:类变量
小写字母或者下划线(_)开头的变量:局部变量
基本语法
Ruby 的赋值不需要加任何关键字,直接一个 变量名=值 即可
变量
x = 25 # => 25x # => 25
注意赋值语句返回了赋的值,这意味着你可以用多重赋值语句
x = y = 10 # => 10x # => 10y # => 10# 除了整数,我们还可以使用booleans、string、symbol、float 等数据类型# booleanstrue_boolean = truefalse_boolean = false# string my_name = 'johnny'# symbola_symbol = :my_symbol# float book_price = 15.80
算术操作
1 + 1 # =>27 - 1 # => 610 * 9 # => 9042 / 7 # => 62 ** 5 # => 32 2的5次方5 % 3 # => 2
位运算符
3 & 5 # => 13 | 5 # => 73 ^ 5 # => 6
打印输出
puts "I'm printing!" # 打印输出,并在末尾加换行符#=> I'm printing!#=> nilprint "I'm printing!" # 打印输出,不加换行符#=> I'm printing! => nil# ⭐除此之外,还可以用缩写 pp "I'm printing!"
注释
# 单行注释=begin 多行注释=end
条件语句:控制流
通过 if...else... 做条件判断。如果为真,它会执行语句中的内容。例如:
if true puts "Hello Ruby If"endif 2 > 1 puts "2 大于 1"end
与 JavaScript 相比,它没有更多的括号,只写必要的东西,但是因为没有中括号,所以当要结束时,需要用 end 来做分离
后续的 def(函数标识符)、class(类标识符)都会有 end 来做结束
除此之外,还有 if...else...:
if 2 > 1 puts "2 大于 1"else puts "2 小于 1"end
还有 elsif 语法,注意不是 elseif,而是elsif,它,缩写了
if 2 > 1 puts "2 大于 1"elsif 2 < 1 puts "2 小于 1"else puts "2 等于 1"end
我们还可以用「倒装句」来写 if 语句
def hey_ho? trueendputs "let's go" if hey_ho?
循环/迭代器
在 Ruby 中,我们可以以多种不同的方式进行迭代。这里我们讨论三个迭代器:while、for 和 each
while 循环
只要语句为真,代码块中的代码就会被执行,如打印 1 到 10 的数字:
num = 1while num <= 10 puts num num += 1end
看看,Ruby 的代码整洁的好看
For 循环
将变量 num 传递给块,for 语句将迭代它:
for num in 1...10 puts numend
Each 迭代器
与其他两种不同,这种写法更像调用方法
[1, 2, 3, 4, 5].each do |num| puts numend
其中 each 迭代器和 for 循环有什么不同呢? each 迭代器只维护迭代块内的变量,而 for 循环允许变量存在于块外
# for vs each# for loopingfor num in 1...5 puts numendputs num # => 5# each iterator[1, 2, 3, 4, 5].each do |num| puts numendput num # => undefined local variable or method `num' for main:Object (NameError)
这里就牵扯到 block(块)了,在用 each 的时候也用到了 do 关键字,它会定义块,块会形成作用域,作用域内部的变量,外部不能方面(这里可以联想到 ES6 中的块级作用域)
数据类型
Ruby 的数据类型有很多,很细分,有String、Number、Float、Array、Hash、Set、Range、Symbol、Boolean、Nil 等等。我们通过 x.class 能得知它的数据类型,通过object_id 得知它的内存地址
String(字符串)
a = 'asdf' # asdf 将 asdf 赋值给aa[0] = 'b' # ruby 中一切皆对象a.object_id # 70123455667 a[1] = 'c' # ca.object_id # 70123455667 ,a 不变b = 'asdf' # 与 a 的值一样b.object_id # 72343954534,但 object_id 不一样,说明他们的内存地址不一样,和 JavaScript 的引用类型一样,每次赋值都存在堆内存里,所以说 ruby 性能差# 在 Ruby 中,一切皆对象,比 JavaScript 更加彻底a.class # String class 方法是判断它的类型
字符串的多种赋值方式
a = 'asdf' # 用单引号"something#{a}" # somethingasdf 双引号相当于 javascript 中的模板字符串(``)%q('asddas'dasda'') # "'asddas'dasda''" 保留你输入的任何值%Q("sadsd") # "\"\"sadsd\"\"" 转义 <<-TextsdsddsadsaddsdasTEXT # 多行"asdfgh".reverse # hgfdsa 反转"hello".include?('o') # true 是否存在字母o"hello".index('l') # 2"asdf".sub('s', 'b') # "adbf" 将 s 替换成 b"asdf".sub!('s', 'b') # "abdf" ! 有什么用呢,它能改变原值a = 'asdf' # asdfa.sub('s', 'b') # abdfa # asdf a 的值没有变化a.sub!('s', 'b') # abdfa # abdf a 的值发生改变a.size # 字符串长度
Integer(整数)/Float(浮点数)
66 # 6666.class # Integer 整数3.2 # 3.23.2.class # Float 浮点数3.even? # 是否为偶数3.odd? # 是否为奇数283.to_s # 转化为 string 3.times { p 'love' } # 打印三次 love3 & 1 # 1 3.232323.round(2) # 小数点保留 2 位
如果想把数字1存储在名为 one 的变量中:
one = 1
同理,如果要赋值数字2,数字1000,就可以如此:
two = 2some_number = 10000
数组
array = [1, 2, 3, 4, 5] # => [1, 2, 3, 4, 5]
数组可以包含不同类型的元素
[1, "hello", false] # => [1, "hello", false]
数组可以被索引,从前面开始
array[0] # => 1array.first # => 1 # 牛逼,这样都可以,array.second 是不是 hello,不妨一试array[12] # => nil # nil 就是 js 中的 null 的意思吧
往数组中添加新值最常见的方法是 push 和 <<
array = []array.push("1")array.push("2")puts array[0] # 1puts array[1] # 2
<< 方法略有不同:
array = []array << "3"array << "4"puts array[0] # 3puts array[1] # 4
当然,还可以把 << 当作方法来使用,我的意思是说可以用. 来调用
array = []array.<<("5")puts array[0] # 5
数组是对象,引用类型
a = [] # []a.object_id # 7012434563b = a # []b.object_id # 7012434563 与 a 一样,引用的是同一个内存地址a << 'foo' # ["foo"]a # ["foo"]a.object_id # 7012434563b # ["foo"]b.object_id # 7012434563# 创建方式: 对象字面量[]、还有 Array.newa = Array.new # []a = Array.new(3) # [nil, nil, nil] a = Array.new(3, 0) # [0,0,0]
特别注意
a = Array.new(3, 'asdf') # ["asdf", "asdf", "asdf"]a[0] # "asdf"a[0][0] = 'b' # ba[0] # "bsdf"a # ["bsdf", "bsdf", "bsdf"]a[0].object_id # 7012838949a[1].object_id # 7012838949a[2].object_id # 7012838949# 因为它们是引用对象,指向同一个引用# 如何只改变数组中的第一项,不改变其他的,使用 block(块)a = Array.new(3) { 'asdf' } # ["asdf", "asdf", "asdf"]a[0].object_id # 7022838393a[1].object_id # 7323856575a[2].object_id # 7342657667
第三种创建数组的方法
arr = %w(foo, bar, baz) # ["foo", "bar", "baz"]
数组的常用方法
# 以上面 arr 为例子arr[0] # "foo"arr[-1] # "baz"arr[1..2] # 取区间["bar", "baz"]arr.fetch(0) # fooarr.fetch(4) # 报错 arr.fetch(4, "asdf") # "asdf"arr.length # 3arr.include?("sdad") # 是否存在sdad ,falsearr.include?('foo') # truearr.empty?() # 是否为空 falsearr.push('ber') # ["foo", "bar", "baz", "ber"]arr[7] = 'asdf' # ["foo", "bar", "baz", "ber", nil, nil, nil, "asdf"]arr.delete("ber") # ["foo", "bar", "baz", nil, nil, nil, "asdf"]arr.push("bar") # ["foo", "bar", "baz", nil, nil, nil, "asdf", "bar"]arr.uniq # 取唯一,删除多余的值 ["foo", "bar", "baz", nil, "asdf"]# arr.uniq! !会改变原数组arr.shuffle # 颠倒 ["asdf", nil, nil, nil, "baz", "bar", "foo"]arr1 = [[1,2,3], [4, 5]]arr.flatten # 扁平化 [1, 2, 3, 4, 5]
数组的遍历方法
arr = [1, -1, 2, 3, -4]arr.each { |e| p e } # e 为数组中的每一项, p 为 put 打印, p e 打印每一项值# 1, -1, 2, 3, -4arr.reverse_each { |e| p e} # 倒着遍历 -4, 3, 2, -1, 1arr.each_width_index {|e, i| p [e, i] } # [1, 0] [-1, 1] [2, 2] [3, 3] [-4, 4]arr.sort # [-4, -1, 1, 2, 3]arr.select { |e| e > 0} # 选择大于0的元素,[1, 2, 3]
哈希表:Key-Value数据结构/字典集合
JavaScript 的数据类型中是没有 hash 的,但在 Ruby 中有。它的特点是以键/值对表示
因为在数组中我们用数字索引来获取到值,而 hash 数据结构中可以使用数字、字符串或者其他类型来做索性
随便一说,JavaScript 中虽然没有 hash,但是它的 object 的功能就是 hash
哈希就键值对的结合,例如:
hash = {"color" => "green", "number" => 5}hash.keys #=> ['color', 'number']# 哈希表可以通过键快速地查询hash['color'] # => 'green'hash['number'] # => 5# 查询一个不存在的键会返回 nilhash['nothing here'] # => nil# 添加数据hash['print'] = 27.6 # => 27.6
{ } 字面量表示 hash
a = { key: 'value' } # {:key => "value"}b = a # {:key => "value"}b.object_id # 91860a.object_id # 91860a[:key] = 'foo' # fooa # {:key => "foo"}b # {:key => "foo"}
另一种创建 hash 的方法,new
Hash.new # {}h = Hash.new(3) # {}h[0] # 3h[1] # 3
哈希的方法
h = {a: 1, b: 2}h[:c] = 3h # {:a=>1, :b=>2, c=>3}h[:a] # 1h.delete(:a) # 1h # {:b=>2, c=>3}h.assoc(:b) # 获取 key 和 value [:b, 2]h.empty?() false h.has_value?(2) # 是否有值 2, trueh.has_key?(:b) # 是否有值:b, trueh.keys # [:b, :c]h.values # [2, 3]h.to_a # 变成 array [[:b, 2], [:c, 3]]h2 = {d: 4}h.merge(h2) # {:b=>2,:c=>3,:d=>4}
hash 的遍历方法
h.each { |key, value| p [key, value]} # [:b, 2] [:c, 3]h.each_key { |key| p key } # :b :ch.each_value {|v| p v} # 2 3h.select { |key| key == :b} # {:b=>2}
集合(Set)
require 'set' # 命令行中默认不引用 seta = Set.new [1, 2] # <Set: {1, 2}>a.add("foo") # Set: {1, 2, "foo"}>b = Set.new [2, 3, 4] #Set: {2, 3, 4}>a & b # Set: {2}>a | b # Set: {1, 2, ”foo", "3", "4"}>a <= b # b 是否是 a 的子集, falseb <= a # a 是否是 b 的子集, false
范围(Range)
闭区间,使用两个点(..)来表示
r = 1..5 # 1, 2, 3, 4, 5 r.include?(2) # truea = [1, 2, 3, 4]a[1..2] # [2, 3]
开区间,使用三个点(...)来表示
r = 1...5 # 包括 1, 2, 3, 4
其他数据类型
当然,还有一些会用但比较简单的数据类型,这里简单带过
Symbol:符号:不可变类型。优点,查找速度快,缺点是不会被垃圾回收,造成内存不够的可能
Boolean:布尔值
Nil:空值
Regexp:正则表达式
函数(方法)
def double(x) x * 2end# 函数(以及所在的块)隐式地返回最后语句的值double(2) # => 4# 当不存在歧义的时候括号可有可无double 3 # => 6double double 3 # => 12def sum(x, y) x + yend# 方法的参数通过逗号分隔sum 3, 4 #=> 7sum sum(3, 4), 5 => 12# yield # 所有的方法都有一个隐式的,可选的块参数# 可以用 ‘yield’ 关键字调用def surround puts "{" yield puts "}"endsurround { puts "hello world" }# {# hello world# }# => nil
面向对象编程
对象
在 Ruby 中,(几乎)所有东西都是对象
# 数字是对象3.class #=> Interger3.to_s #=> "3"# 字符串是对象"Hello".class #=> String# 方法也是对象"Hello".method(:class).class => Method
类(Class)
类 = 属性 + 方法
在前面,我们谈到了 Ruby 的对象,我们说 Ruby 中的任何东西都是对象。而作为一种面向对象的编程语言,Ruby 使用了类和对象的概念
“类”是一种定义对象的方法,如图书、狗、自行车
“对象”有两个主要特征:数据(属性)和行为(方法)
语法很简单,例如:
class Bookend
我们用 class 语句定义 Book 并以 end 结束
对象是类的实例。我们通过调用 .new 方法创建一个实例
book = Book.new
这里的书 是书类的 一个对象(或实例)
我们定义书类将具有 4 个属性:书名、作者、星、价格
我们重新定义我们的类 Book :
class Book def initialize(title, author, star, price) @title = title @author = author @star = star @price = price endend
我们将 initialize 方法成为构造方法,当我们要使用这个类时,就可以这样:
book1 = Book.new('金瓶梅', '兰陵笑笑生', '五星', 16.8)
我们还可以定义书的行为:读书
class Book def initialize(title, author, star, price) @title = title @author = author @star = star @price = price end def read puts "我读 #{@title},这本书的作者是 #{@author},推荐指数 #{@star},价格 #{@parice} 元" endend
使用该类创建对象的实例代码如下:
book1 = Book.new('金瓶梅', '兰陵笑笑生', '五星', 16.8)book2 = Book.new('南回归线', '亨利·米勒', '五星', 66.6)book1.read # 我读金瓶梅,这本书的作者是兰陵笑笑生,推荐指数五星,价格 16.8 元book12.read # 我读南回归线,这本书的作者是亨利·米勒,推荐指数五星,价格 66.6 元
参考资料
Y分钟速成X
Learning Ruby: From Zero to Hero
⚠️「随朱波流」只是记录本人关于编程、生活、个人成长、思考的自留地。如有需要,可前往 https://azhubaby.com(阅读原文)中查看第一手文章