写在前面

我们当然希望能在本地中执行代码,如果你还没有只是看看,还没安装 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 # => 25
x # => 25
注意赋值语句返回了赋的值,这意味着你可以用多重赋值语句
x = y = 10 # => 10
x # => 10
y # => 10

# 除了整数,我们还可以使用booleans、string、symbol、float 等数据类型
# booleans
true_boolean = true
false_boolean = false

# string 
my_name = 'johnny'

# symbol
a_symbol = :my_symbol

# float 
book_price = 15.80

算术操作

1 + 1 # =>2
7 - 1 # => 6
10 * 9 # => 90
42 / 7 # => 6
2 ** 5 # => 32 2的5次方
5 % 3 # => 2

位运算符

3 & 5 # => 1
3 | 5 # => 7
3 ^ 5 # => 6

打印输出

puts "I'm printing!" # 打印输出,并在末尾加换行符
#=> I'm printing!
#=> nil

print "I'm printing!" # 打印输出,不加换行符
#=> I'm printing! => nil

# ⭐除此之外,还可以用缩写 p
p "I'm printing!"

注释

# 单行注释
=begin
    多行注释
=end

条件语句:控制流

通过 if...else... 做条件判断。如果为真,它会执行语句中的内容。例如:

if true
    puts "Hello Ruby If"
end

if 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?
    true
end

puts "let's go" if hey_ho?

循环/迭代器

在 Ruby 中,我们可以以多种不同的方式进行迭代。这里我们讨论三个迭代器:while、for 和 each

while 循环

只要语句为真,代码块中的代码就会被执行,如打印 1 到 10 的数字:

num = 1

while num <= 10
    puts num
    num += 1
end

看看,Ruby 的代码整洁的好看

For 循环

将变量 num 传递给块,for 语句将迭代它:

for num in 1...10
    puts num
end

Each 迭代器

与其他两种不同,这种写法更像调用方法

[1, 2, 3, 4, 5].each do |num|
    puts num
end

其中 each 迭代器和 for 循环有什么不同呢? each 迭代器只维护迭代块内的变量,而 for 循环允许变量存在于块外

# for vs each

# for looping
for num in 1...5
    puts num
end

puts num # => 5

# each iterator
[1, 2, 3, 4, 5].each do |num|
    puts num
end

put 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 赋值给a
a[0] = 'b' # ruby 中一切皆对象
a.object_id # 70123455667 
a[1] = 'c' # c
a.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\"\"" 转义 
<<-Text
sdsd
dsadsad
dsdas
TEXT # 多行
"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' # asdf
a.sub('s', 'b') # abdf
a # asdf  a 的值没有变化
a.sub!('s', 'b') # abdf
a # abdf  a 的值发生改变
a.size # 字符串长度

Integer(整数)/Float(浮点数)

66 # 66
66.class # Integer 整数
3.2 # 3.2
3.2.class # Float 浮点数
3.even? # 是否为偶数
3.odd? # 是否为奇数
283.to_s # 转化为 string 
3.times { p 'love' } # 打印三次 love
3 & 1 # 1 
3.232323.round(2) # 小数点保留 2 位

如果想把数字1存储在名为 one 的变量中:

one = 1

同理,如果要赋值数字2,数字1000,就可以如此:

two = 2
some_number = 10000

数组

array = [1, 2, 3, 4, 5] # => [1, 2, 3, 4, 5]

数组可以包含不同类型的元素

[1, "hello", false] # => [1, "hello", false]

数组可以被索引,从前面开始

array[0] # => 1
array.first # => 1 # 牛逼,这样都可以,array.second 是不是 hello,不妨一试
array[12] # => nil # nil 就是 js 中的 null 的意思吧

往数组中添加新值最常见的方法是 push 和 <<

array = []
array.push("1")
array.push("2")
puts array[0] # 1
puts array[1] # 2

<< 方法略有不同:

array = []
array << "3"
array << "4"
puts array[0] # 3
puts array[1] # 4

当然,还可以把 << 当作方法来使用,我的意思是说可以用. 来调用

array = []
array.<<("5")

puts array[0] # 5

数组是对象,引用类型

a = [] # []
a.object_id # 7012434563
b = a # []
b.object_id # 7012434563 与 a 一样,引用的是同一个内存地址
a << 'foo' # ["foo"]
a # ["foo"]
a.object_id # 7012434563
b # ["foo"]
b.object_id # 7012434563

# 创建方式: 对象字面量[]、还有 Array.new
a = 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' # b
a[0] # "bsdf"
a # ["bsdf", "bsdf", "bsdf"]
a[0].object_id # 7012838949
a[1].object_id # 7012838949
a[2].object_id # 7012838949
# 因为它们是引用对象,指向同一个引用
# 如何只改变数组中的第一项,不改变其他的,使用 block(块)
a = Array.new(3) { 'asdf' } # ["asdf", "asdf", "asdf"]
a[0].object_id # 7022838393
a[1].object_id # 7323856575
a[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) # foo
arr.fetch(4) # 报错 
arr.fetch(4, "asdf") # "asdf"
arr.length # 3
arr.include?("sdad") # 是否存在sdad ,false
arr.include?('foo') # true
arr.empty?() # 是否为空 false
arr.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, -4
arr.reverse_each { |e| p e} # 倒着遍历 -4, 3, 2, -1, 1
arr.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

# 查询一个不存在的键会返回 nil
hash['nothing here'] # => nil

# 添加数据
hash['print'] = 27.6 # => 27.6

{ } 字面量表示 hash

a = { key: 'value' } # {:key => "value"}
b = a # {:key => "value"}
b.object_id # 91860
a.object_id # 91860
a[:key] = 'foo' # foo
a # {:key => "foo"}
b # {:key => "foo"}

另一种创建 hash 的方法,new

Hash.new # {}
h = Hash.new(3) # {}
h[0] # 3
h[1] # 3

哈希的方法

h = {a: 1, b: 2}
h[:c] = 3
h # {:a=>1, :b=>2, c=>3}
h[:a] # 1
h.delete(:a) # 1
h # {:b=>2, c=>3}
h.assoc(:b) # 获取 key 和 value [:b, 2]
h.empty?() false 
h.has_value?(2) # 是否有值 2, true
h.has_key?(:b) # 是否有值:b, true
h.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 :c
h.each_value {|v| p v} # 2 3
h.select { |key| key == :b} # {:b=>2}

集合(Set)

require 'set' # 命令行中默认不引用 set
a = 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 的子集, false
b <= a # a 是否是 b 的子集, false

范围(Range)

闭区间,使用两个点(..)来表示

r = 1..5 # 1, 2, 3, 4, 5 
r.include?(2) # true
a = [1, 2, 3, 4]
a[1..2] # [2, 3]

开区间,使用三个点(...)来表示

r = 1...5 # 包括 1, 2, 3, 4

其他数据类型

当然,还有一些会用但比较简单的数据类型,这里简单带过

Symbol:符号:不可变类型。优点,查找速度快,缺点是不会被垃圾回收,造成内存不够的可能

Boolean:布尔值

Nil:空值

Regexp:正则表达式

函数(方法)

def double(x)
    x * 2
end

# 函数(以及所在的块)隐式地返回最后语句的值
double(2) # => 4

# 当不存在歧义的时候括号可有可无
double 3 # => 6

double double 3 # => 12

def sum(x, y)
    x + y
end

# 方法的参数通过逗号分隔
sum 3, 4 #=> 7

sum sum(3, 4), 5 => 12

# yield 
# 所有的方法都有一个隐式的,可选的块参数
# 可以用 ‘yield’ 关键字调用

def surround
    puts "{"
    yield
    puts "}"
end

surround { puts "hello world" }

# {
# hello world
# }
# => nil

面向对象编程

对象

在 Ruby 中,(几乎)所有东西都是对象

# 数字是对象
3.class #=> Interger
3.to_s #=> "3"

# 字符串是对象
"Hello".class #=> String

# 方法也是对象
"Hello".method(:class).class => Method

类(Class)

类 = 属性 + 方法

在前面,我们谈到了 Ruby 的对象,我们说 Ruby 中的任何东西都是对象。而作为一种面向对象的编程语言,Ruby 使用了类和对象的概念

“类”是一种定义对象的方法,如图书、狗、自行车

“对象”有两个主要特征:数据(属性)和行为(方法)

语法很简单,例如:

class Book
end

我们用 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
    end
end

我们将 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} 元"
    end
end

使用该类创建对象的实例代码如下:

book1 = Book.new('金瓶梅', '兰陵笑笑生', '五星', 16.8)
book2 = Book.new('南回归线', '亨利·米勒', '五星', 66.6)

book1.read # 我读金瓶梅,这本书的作者是兰陵笑笑生,推荐指数五星,价格 16.8 元
book12.read # 我读南回归线,这本书的作者是亨利·米勒,推荐指数五星,价格 66.6 元

参考资料

系列文章


山头人汉波
391 声望554 粉丝