数组

Ruby中的数组是一个容器,数组中的每个元素都是一个对象的引用

注意,Array类中包含了Enumerable模块,所以Enumerable中的方法也都能使用,例如Enumerable中的reduce()方法也是非常好用的方法。

创建数组

字面常量创建

# 1.使用[xxx]方式创建
arr1 = ["Perl", "Python", "Ruby"]

# 2.空数组
arr = []

# 3.使用%w或%W可以省略引号和逗号,而使用空格分隔
# %w(小写)不解释内插表达式,%W(大写)解释内插
# 此时中括号可以换成其它符号,如%{}、%<>、%##
# 如果元素中要保留空格,使用反斜线转义
arr2 = %w[Perl Python Ruby]   # 3个元素
arr3 = %w<Perl Python\ Ruby>  # 2个元素

a = "Perl"
arr4 = %w(#{a} Python Ruby)   # #{a}不做变量替换,第一个元素为`\#{a}`
arr5 = %W(#{a} Python Ruby)   # #{a}会做变量替换

# 4.使用%i创建符号symbol数组
arr6 = %i(Perl Python Ruby)  # [:Perl, :Python, :Ruby]

Array.new()创建

Array类的new()方法可以创建数组。

ary = Array.new    #=> []
Array.new(3)       #=> [nil, nil, nil]
Array.new(3, true) #=> [true, true, true]

当new()指定了第二个参数时,各初始化的元素将指向同一个对象

arr=Array.new(3, "abc")
arr[1][0]="A"   #=> ["Abc", "Abc", "Abc"]

所以,建议初始化的元素采用不可变对象,如数值、symbol、bool的true和false。或者使用下面这种块结构的初始化方式。

可以将块结构接管Array.new的第二个默认值参数,它将每个元素的索引值传递到块变量中。例如:

Array.new(4) {Hash.new}    #=> [{}, {}, {}, {}]
Array.new(4) {|i| i.to_s } #=> ["0", "1", "2", "3"]

例如,初始化互不影响的3个"abc"元素数组。

arr = Array.new(3) {"abc"}
arr[0][1]="A"
p arr          # ["aAc", "abc", "abc"]

多维数组:

Array.new(3){Array.new(2)} #=> [[nil, nil], [nil, nil], [nil, nil]]

Array()创建

通过Kernel模块提供的Array(arg)方法创建数组。

该方法将给定参数arg转换成数组。

Array({:a => "a", :b => "b"}) #=> [[:a, "a"], [:b, "b"]]
Array(1..5)  #=> [1, 2, 3, 4, 5]

to_a()创建

如果某个类定义了to_a()方法,可以将对象转换成数组结构。例如,hash类有to_a()方法,所以可以将hash结构转换成数组。

my_hash = {Perl: "Larry Wall", Ruby: "Matz"}
arr = my_hash.to_a()  # [[:Perl, "Larry Wall"],[:Ruby, "Matz"]]

split()从字符串创建

字符串对象有split()方法,可以按照指定的分隔符将字符串切割成数组。

p "perl python ruby".split  # ["perl", "python", "ruby"]

访问数组元素

索引/slice取元素

获取数组的方式有多种。列出了以下几种常见的(返回多个元素的方式将以新数组的方式返回):

  • arr[1]或arr.at(n)或arr.slice(n):获取index=1的元素
  • arr[1..3]或arr.slice(1..3):获取index=1、2、3的3个元素
  • arr[1...3]或arr.slice(1...3):获取index=1、2的两个元素
  • arr[2, 3]或arr.slice(2, 3):获取从index=2开始的3个元素,即index=2、3、4的元素
  • arr[-1]:获取最后一个元素
  • arr[-3]:获取倒数第三个元素

其中,at()方法和slice()方法很少用,因为arr[]的方式已经足够。

如果索引越界,则返回nil。但是有一个特殊情况,当使用slice的操作时,因为要返回的是数组,所以有如下"异常"情况:

a = %w(a b c d e)
a[4]       #=>"e"
a[4,0]     #=>[]
a[4,1]     #=>["e"]
a[4..100]  #=>["e"]

a[5]       #=>nil
a[5,0]     #=>[]
a[5,1]     #=>[]
a[5..100]  #=>[]

a[6]       #=>nil
a[6,0]     #=>nil
a[6,1]     #=>nil

上面a[5]返回nil,但a[5,x]返回空数组,而a[6]以及a[6,x]则直接返回nil。首先a[6]以及a[6,x]都是索引越界,所以返回nil。而a[5]的5是索引越界,所以返回nil。问题是a[5,x]返回的是空数组,而不是nil。这是因为a[5,x]是一个slice操作,它要求返回新数组。从下面的slice操作可以理解为什么a[5,1]返回空数组。

p a[0, 5]   # ["a", "b", "c", "d", "e"]
p a[1, 4]   # ["b", "c", "d", "e"]
p a[2, 3]   # ["c", "d", "e"]
p a[3, 2]   # ["d", "e"]
p a[4, 1]   # ["e"]
p a[5, 0]   # []
p a[5, 1]   # []

也就是说,对于slice操作,arr[len, x]的arr[len]可以认为是最边缘的元素,尽管arr[len]已经越界了。

其实slice()是按规则删除元素,只不过它不影响原始数组。而slice!()则是删除一些元素并直接影响原始数组,也就是在原处修改数组。

a = [1, 2, 3, 4]
a.slice!(1,2)  # 返回:[2, 3],a变成[1, 4]

Array.values_at()取分散元素

Array类提供了values_at()方法,可以取得每个数组对象中指定索引处的多个分散元素。

例如:

a = %w(a b c d e)
p a.values_at(1)          # ["b"]
p a.values_at(0, 0, 2, 4) # ["a", "a", "c", "e"]
p a.values_at(1, 2, 10)   # ["b", "c", nil]
p a.values_at(1..3)       # ["b", "c", "d"]

除了这种方式可以取得分散元素,还可以使用map()函数进行操作。例如,等价于values_at(0,0,2,4,10)的写法为:

p [0, 0, 2, 4, 10].map{|i| a[i]}

其它一些方法

比如,默认情况下索引越界时将返回nil,使用fetch()方法可以获取指定索引的元素,且可以指定越界时的默认值,否则直接报错。

a=%w(a b c d e)
p a.fetch(10)  # 报错:索引越界IndexError
p a.fetch(10, "ten") # 指定默认值

使用first()和last()可以取得数组中的第一个元素和最后一个元素,它们会取得元素后立即退出。

a = %w(a b c d e)
p a.first     # "a"
p a.last      # "e"

使用take()可以取得数组中的前n个元素,使用drop()可以取得除了数组前n个元素外剩下的元素。它们都不会删除元素。

a = %w(a b c d e f)
p a.take(2)  # ["a", "b"]
p a.drop(2)  # ["c", "d", "e", "f"]
p a          # ["a", "b", "c", "d", "e", "f"]

为数组元素赋值

[] at slice方法都可以为数组中指定位置处元素赋值。唯一需要注意的是范围赋值操作,将其认为是将范围内的值设置为新值。

a = %w(a b c d e)

# 单元素赋值
a[1] = "bb"  # 为第2个元素赋值

# 范围赋值
a[1, 2] = ["bb", "cc"]  # a=%w(a bb cc d e)
a[1, 2] = %w(bb)  # a=%w(a bb d e)
a[1, 2] = %w(bb cc dd)  # a=%w(a bb cc dd d e)

# 完全插入元素:指定len为0
a[1, 0] = %w(B C) # a=%w(a B C b c d e)
a[1..1] = %w[B C] # 与上式等价

# 完全删除元素:将右边对象设置为空数组
a[1, 2] = []  # a=%w(a d e)

赋值语句的返回值是所赋值内容。例如a[1,0] = %w[B C]的返回值是%w[B C],于此同时原始数组被修改。

扩展数组对象

数组可以执行+ - * & |操作,不过- & |是将数组作为集合进行操作的,相关内容见后文对应小节。而使用+*可以扩展数组。

因为ruby中的一元运算符操作x += y和二元运算符操作x = x + y完全等价,都会创建新对象x。所以,涉及到大数组时,可能效率会比较低下。

+可以将两数组加在一起返回一个新数组:

arr = [1, 2, 3]
arr1 = arr + [3, 4, 5]  # arr不变,arr1=[1,2,3,3,4,5]

concat()方法也能将0到多个数组扩展到一个数组上。注意,concat()会将参数数组全都"压平"然后追加到原数组尾部:

["a", "b"].concat(["c", "d"])   # ["a", "b", "c", "d"]
["a"].concat(["b"], ["c", "d"]) # ["a", "b", "c", "d"]
["a"].concat     # ["a"]

a = [1, 2, 3]
a.concat([4, 5])  # a=[1, 2, 3, 4, 5]

a = [1, 2]
a.concat(a, a)   # a=[1, 2, 1, 2, 1, 2]

*对于数组有两种用法:

# 1.数组乘一个数值对象,返回新数组
# 数值必须在数组后面,不能放前面
arr = [1, 2, 3]
arr1 = arr * 2  # arr不变,arr1=[1,2,3,1,2,3]

# 2.数组乘一个字符串,将返回字符串而非新数组
arr = [1, 2, 3]
arr1 = arr * ","   # arr不变,arr1="1,2,3"
arr2 = arr * ",+"  # arr不变,arr2="1,+2,+3"

数组信息和数组测试

获取数组长度,可以使用count()、length()或size()方法,后两者等价。而count()方法通过参数或语句块的方式还能获取满足条件的元素个数。

a = %w(a b c d e)
p a.length      # 5
p a.count       # 5
p a.size        # 5

ary = [1, 2, 4, 2]
ary.count                  # 4,数组元素个数
ary.count(2)               # 2,等于2的元素个数
ary.count {|x| x%2 == 0}   # 3,偶元素个数

检查数组是否为空数组,使用empty?()方法:

p a.empty?      # false

检查数组是否包含某元素,使用include?()方法:

p a.include?('c')  # true
p a.include?('x')  # false

向数组中增、删元素

插入元素

push()/append()、unshift()、insert()或特殊符号<<,注意它们都是原处修改数组对象的:

# push()向尾部插入元素
# push()返回数组自身,可以链式插入
# append()等价于push()
arr = [1, 2, 3, 4]
arr.push(5)   # arr = [1, 2, 3, 4, 5]
arr.push(6).push(7) #  arr = [1,2,3,4,5,6,7]

# <<向尾部插入元素
# <<返回数组自身,所以可以进行链式插入
arr = [1, 2, 3, 4]
arr << 5          # arr = [1, 2, 3, 4, 5]
arr << 6 << 7     # arr = [1,2,3,4,5,6,7]
arr <<8 <<[9, 10] # [1,2,3,4,5,6,7,8,[9,10]]

# unshift()向头部插入元素
arr = [1, 2, 3, 4]
arr.unshift(0) # arr = [0, 1, 2, 3, 4]

# insert()向给定索引位置处插入元素,可一次性插入多个
arr = [0, 1, 2, 3, 4]
arr.insert(3, "Ruby") # [0, 1, 2, 'Ruby', 3, 4]
arr.insert(3,"Perl","Shell") # # [0,1,2,'Perl','Shell','Ruby',3,4]

# 通过范围赋值的方式插入元素
arr = [0, 1, 2, 3, 4]
arr[1,0] = [11, 22]  # [0, 11, 22, 1, 2, 3, 4]
arr[1..1] = [11, 22]  # 与上等价

删除元素

按位置删除元素

pop()、shift()、delete_at():

# pop()移除数组尾部元素并返回该元素
arr = [1, 2, 3, 4, 5, 6]
arr.pop   # 返回6, arr变成[1, 2, 3, 4, 5]

# shift()从数组头部移除一个元素并返回该元素
arr.shift     # 返回1,arr变成[2, 3, 4, 5]

# delete_at()移除给定索引位置处的元素并返回该元素
arr.delete_at(2)  # 返回4,arr变成[2, 3, 5]

# 通过范围赋值的方式删除元素
arr = [1, 2, 3, 4, 5, 6]
arr[1, 2] = []   # arr = [1, 4, 5, 6]

按给定值删除元素

delete():

# delete()删除数组中等于某个值的元素
arr1=[1, 2, 2, 4]
arr2 = %w(Perl Shell Python Ruby)
arr1.delete(2)       # 返回2,arr1变成[1,4]
arr2.delete "Shell"  # 返回"Shell",arr2变成%w(Perl Python Ruby)

删除重复元素

uniq()、compact()以及成对的带感叹号后缀的uniq!()、compact!():

# uniq()删除重复元素,不是原处修改对象的
arr=%w(a b c x y a c b y)
uniq_arr = arr.uniq   # uniq_arr=%w(a b c x y),arr不变

# uniq!()删除重复元素,直接修改原始对象且返回修改后的数组
arr=%w(a b c x y a c b y)
uniq_arr = arr.uniq!   # uniq_arr和arr都变成%w(a b c x y)
# compact()删除所有nil元素,压缩数组,不是原处修改对象
arr = ["a", "b", "c", nil, "a", "c", nil, "b"]
compact_arr = arr.compact
p compact_arr    # ["a", "b", "c", "a", "c", "b"]
p arr            # ["a","b","c",nil,"a","c",nil,"b"]

# compact!()删除所有nil元素,压缩数组,原处修改对象
arr = ["a", "b", "c", nil, "a", "c", nil, "b"]
compact_arr = arr.compact!
p compact_arr    # ["a", "b", "c", "a", "c", "b"]
p arr            # ["a", "b", "c", "a", "c", "b"]

清空数组

clear()直接清空数组:

a = [1,2,3]
a.clear()    # a=[]

迭代数组

迭代方式有很多,通过for、while、times、数组自带的each、each_index、reverse_each,还有Mix-in Enumerator后获取的迭代方式:each_cons、each_slice、each_entry、each_with_index、with_index、each_with_object等。

此处仅简单介绍for/while/times迭代和数组自身几个迭代方法,关于从Enumerator获取的相关的迭代方式,参见对应文章:Enumerator各种迭代方式

数组自身迭代方法

for、while、times迭代

# for
x = %w(a b c d e)

for i in x
  puts "element: #{i}"
end

# while
x = %w(a b c d e)

i=0
while i < x.length
  puts "element: #{x[i]}"
  i += 1
end

# times
x = %w(a b c d e)

x.length.times do |n|
  puts "element: #{x[n]}"
end

each()

each {|item| block} → ary
each → Enumerator

迭代数组中每个元素,并将每个元素传递给语句块中的变量。最后返回数组自身。

# each迭代数组,不会修改原始数组
arr = [1, 2, 3, 4, 5]
arr.each {|a| print a -= 10, " "}
## 输出:-9 -8 -7 -6 -5
## arr=[1, 2, 3, 4, 5]

each_index()

each_index {|index| block} → ary
each_index → Enumerator

迭代数组中的每个元素,并将每个元素的索引传递给语句块中的变量。最后返回数组自身。

a = ["a", "b", "c", "d"]

a.each_index do |x|
  puts "index: #{x}"
end

## 输出结果:
=begin
index: 0
index: 1
index: 2
index: 3
=end

reverse_each()

# reverse_each反序迭代数组
arr = [1, 2, 3, 4, 5]
arr.reverse_each {|a| print "#{a}-"}
## 输出5-4-3-2-1-

数组转换和测试

类方法:

  • Array.try_convert(arg):尝试将arg转换成数组,如果能转换则返回转换后的数组对象,否则返回nil。可以用它来测试参数对象是否是数组
  • to_a():返回self,对于数组自身来说,和to_ary()基本没区别
  • to_ary():返回self,对于数组自身来说,和to_ary()基本没区别
  • to_s():将数组转换成字符串格式
  • to_h():将数组转换成hash类型
  • inspect():等价于to_s()
# try_convert(arg)将转换成数组
arr = Array.try_convert([1])    # [1]
arr1 = Array.try_convert(1)     # nil
arr2 = Array.try_convert("1")   # nil

# 测试元素是否是数组、字符串
if tmp=Array.try_convert(arg)
  # arg is an array
elsif tmp = String.try_convert(arg)
  # arg is a string
end
# to_a()和to_ary(),都返回self
# 表示返回当前数组的引用
# 对于数组自身来说,这两者没差别
arr = ["foo", "bar"]
arr1 = arr.to_a

p arr.eql?(arr1)    # true
p arr.object_id     # 42285320
p arr1.object_id    # 42285320
# to_s()等价于inspect()
# 将数组转换成字符串
arr = ["foo", "bar"]
arr.to_s   # [\"foo\", \"bar\"]
# to_h()将数组转换成hash
h1 = [["foo",1], ["bar", 2]].to_h

# to_h的语句块形式,每个元素都将作为块中的变量
h2 = [["foo", 1],["bar", 2]].to_h {|x| x}
h3 =  ["foo", "bar"].to_h {|s| [s, s * 2] }

p h1   # {"foo"=>1, "bar"=>2}
p h2   # {"foo"=>1, "bar"=>2}
p h3   # {"foo"=>"foofoo", "bar"=>"barbar"}

数组大小、等同的比较

三种比较方式:等值比较==、大小比较<=>和是否同一数组对象的比较eql?()

==

只有两数组长度相同、两数组对应索引位置的元素使用==测试也相等,才判定两数组相等。

返回true/false。注意,如果含有nil元素,则nil与nil的比较结果为true。

[ "a", "c" ]    == [ "a", "c", 7 ]     # false
[ "a", "c", 7 ] == [ "a", "d", "f" ]   # false
[ "a", "c", 7 ] == [ "a", "c", 7 ]     # true
["a", 1, 1.2] == ["a", 1.0, 1.2]       # true

eql?()

只有两数组对象的内容完全相同时返回true,否则返回false。

它是根据各对象计算出的hash值比较的,一般来说比==要严格一点。

[1, 2] == [1, 2.0]      # true
[1, 2].eql?([1, 2.0])   # false
[1, 2].eql?([1, 2])     # true

[1, 2].hash    # 2027605168499122706
[1, 2.0].hash  # 3393919734812826294

equal?()

继承自Object类,它比较的是两者是否是同一对象。这个方法基本上不会被子类重写。

这个方法比==eql?()都要严格。

[1,2].equal?([1,2])  # false

a = [1, 2]
b = a
a.equal?(b)  # true

<=>

这个比较符号,当执行的是obj1 <=> obj2时:

  • obj1大于obj2时,该比较表达式返回1
  • obj1等于obj2时,该比较表达式返回0
  • obj1小于obj2时,该比较表达式返回-1

对于数组而言,对数组中的每个元素都一一对应的比较,直到找到某个数组中的元素不等于另一个数组中对应索引处的元素就返回结果。如果数组长短不一致,且较短数组的所有元素都和另一数组对应元素相等,则长数组更大。于是可以推断,只有两数组长度、内容完全相等(通过==号比较,例如1等于1.0),数组才相等。

arr = [1, 3, 5, 7]
arr1 = [1, 3.0, 5, 7]
arr2 = [1, 3, 5]
arr3 = [1, 3, 5, 7, 9]
arr4 = [1, 3, 7, 5]
arr5 = [1, 3, 7, 5, nil]

p arr <=> arr1   # 0
p arr <=> arr2   # 1
p arr <=> arr3   # -1
p arr <=> arr4   # -1
p arr <=> arr5   # -1

Ruby数组(1):基本用法的更多相关文章

  1. Ruby数组

    Ruby数组是有序的,任何对象的整数索引的集合.每个数组中的元素相关联,并提取到的一个索引.下标与C或Java相似,从0开始.负数索引假设数组末尾,也就是说-1表示最后一个元素的数组索引,-2是数组中 ...

  2. 雷林鹏分享:Ruby 数组(Array)

    Ruby 数组(Array) Ruby 数组是任何对象的有序的.整数索引的集合.数组中的每个元素都与一个索引相关,并可通过索引进行获取. 数组的索引从 0 开始,这与 C 或 Java 中一样.一个负 ...

  3. jquery下json数组的操作用法实例

    jquery下json数组的操作用法实例: jquery中操作JSON数组的情况中遍历方法用的比较多,但用添加移除这些好像就不是太多了. 试过json[i].remove(),json.remove( ...

  4. (转)轻松掌握shell编程中数组的常见用法及示例

    缘起:在老男孩进行linux培训shell编程教学中,发现不少水平不错的网友及同学对数组仍然很迷糊,下面就给大家分享下数组的用法小例子,希望能给大家一点帮助.其实SHELL的数组很简单,好用.我们学习 ...

  5. Ruby数组方法整理

    数组方法整理 方法列表: all().any().none()和one():测试数组中的所有或部分元素是否满足给定条件.条件可以是语句块中决定,也可以是参数决定 append():等价于push() ...

  6. JS数组的基本用法

    JS数组的用法包括创建.取值赋值.添加以及根据下标(包括数值或字符)来移除元素等等,在本文中将为大家详细介绍,感兴趣的朋友可以参考下. 1.创建数组: //1.1直接创建一个数组对象 var arra ...

  7. ruby数组操作方法汇总

    1.数组定义 arr1 = [] arr2 = Array.new arr3 = ['1','2','3'] 2.输出 print arr3,"\n" #123 puts arr3 ...

  8. ruby 数组array 排序sort 和sort!

    1. sort → new_ary click to toggle source sort { |a, b| block } → new_ary Returns a new array created ...

  9. numpy学习笔记 - numpy数组的常见用法

    # -*- coding: utf-8 -*- """ 主要记录代码,相关说明采用注释形势,供日常总结.查阅使用,不定时更新. Created on Mon Aug 20 ...

随机推荐

  1. 匿名函数 javascript

    匿名函数: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF ...

  2. hadoop2-MapReduce详解

    本文是对Hadoop2.2.0版本的MapReduce进行详细讲解.请大家要注意版本,因为Hadoop的不同版本,源码可能是不同的. 以下是本文的大纲: 1.获取源码2.WordCount案例分析3. ...

  3. Vue.set() this.$set()引发的视图更新思考

    引文 vue文档列表渲染中有条注意事项: 这里提到的两种情况实际改变了数据但是没有触发视图更新. 由此引出Vue.set(),先上文档API: this.$set()和Vue.set()本质方法一样, ...

  4. HTML入门13

    构建表格 使用colspan和rowspan添加无单位的数字值作为属性来实现行合并和列合并: <col>来定义列的样式,每一个<col>都会制定每列的样式,对于不需要指定列的样 ...

  5. java代码编译与C/C++代码编译的区别

    Java编译原理 1.Java编译过程与c/c++编译过程不同 Java编译程序将java源程序编译成jvm可执行代码--java字节码. Java在编译过程中一般会按照以下过程进行: (1)JDK根 ...

  6. vue单页面应用刷新网页后vuex的state数据丢失的解决方案

    1. 产生原因其实很简单,因为store里的数据是保存在运行内存中的,当页面刷新时,页面会重新加载vue实例,store里面的数据就会被重新赋值. 2. 解决思路一种是state里的数据全部是通过请求 ...

  7. Python课程学习总结

    Python的介绍 Python是一种高级动态.完全面向对象的语言,函数.模块.数字.字符串都是对象,并且完全支持继承.重载.派生.多继承,有益于增强源代码的复用性. Python是一种计算机程序设计 ...

  8. java小练习

    打印99乘法表 因为有9行9列,所有要用两个for循环 int m; for (int i = 1; i < 10; i++) { for (int j = 1; j <= i; j++) ...

  9. 一个自己研究出来的字符串匹配算法-k子串算法

    前言 最近工作中需要写一个算法,而写完这个算法我却发现了一个很有意思的事情.需要的这个算法是这样的:对于A,B两个字符串,找出最多K个公共子串,使得这K个子串长度和最大.百度之没有这样的算法,然后就开 ...

  10. 微信小程序实现图片是上传、预览功能

    本文实例讲述了微信小程序实现图片上传.删除和预览功能的方法,分享给大家供大家参考,具体如下: 这里主要介绍一下微信小程序的图片上传图片删除和图片预览 1.可以调用相机也可以从本地相册选择 2.本地实现 ...