Ruby中Hash的11个问题解答

最近在学习Rails,因为学习的java为入门语言.些许的不太适应.特别是Ruby的Hash.不过细细的想想就通了.基本上都是通的Key-Value.除了刚开始不太适应写法没什么大的问题.这个是我在网上看到的一篇日志.

1. 如何创建Hash?

代码如下:

x = Hash.new
x = {}
x = {:a => 1, :b => 2}

这些都是创建Hash的方法。
第一个和第二个一样,表示创建空的Hash.
第三个表示创建的Hash,有2对键/值。

2. Hash的键是哪些类型?

可以是任何目标,如:

代码如下:

irb(main):002:0> x={}
=> {}
irb(main):003:0> x[1]=2
=> 2
irb(main):004:0> x["a"]="b"
=> "b"
irb(main):005:0> x[:z] = 100
=> 100
irb(main):006:0> x[[1,2,3]] = [4,5,6]
=> [4, 5, 6]
irb(main):007:0> x
=> {"a"=>"b", [1, 2, 3]=>[4, 5, 6], 1=>2, :z=>100}

有两点注意:
(1)任何目标包括数组、Hash都可以是Key。
(2)使用字符作为Key时,最好使用对应的符号。例如,键"a"可以用:a来代替。这是因为每个字符"a"都是一个目标,而:a是唯一的,

更节省内存。

3. 如何给Hash赋默认的值?

代码如下:

irb(main):003:0> x=Hash.new([])
=> {}
irb(main):004:0> x[:a]
=> []

表示创建一个Hash,它的默认值是空的数组[]。
然后访问x[:a],虽然这个键没有事先定义,但还是有默认值。

请注意:这种方法赋的默认值,都指向同一个目标,这个目标改变了,会影响到所有默认值。

代码如下:

irb(main):005:0> x[:a] << 1
=> [1]
irb(main):006:0> x[:b]
=> [1]

要做到每个键都对应唯一的默认值,应该如此创建Hash:

代码如下:

irb(main):007:0> y=Hash.new do |h,k| h[k] =[] end
=> {}

让我们再看看:

代码如下:

irb(main):009:0> y[:a] << 1
=> [1]
irb(main):010:0> y[:b]
=> []

一个值的改变,不会影响到其他默认值。

4. 如何遍历Hash?


代码如下:

irb(main):018:0* x={1=>2,3=>4,5=>6}                 
=> {5=>6, 1=>2, 3=>4}

irb(main):019:0> x.each do |k,v| puts "#{k} #{v}" end  
5 6
1 2
3 4
=> {5=>6, 1=>2, 3=>4}

使用Hash的each方法。

5. 如何遍历Hash的Key和Value?


代码如下:

irb(main):020:0> x.keys.each do |k| puts k end

irb(main):021:0> x.values.each do |v| puts v end

使用Hash的keys和values方法,它们返回一个数组。

当然还可以使用each_key和each_value方法,顾名思义就是遍历key和value.

代码如下:

irb(main):016:0> x.each_key do |s| puts s end
1
3
=> {1=>2, 3=>4}
irb(main):017:0> x.each_value do |s| puts s end   
2
4
=> {1=>2, 3=>4}

6. Hash可以sort吗?

ruby的hash实际是可以sort的,返回一个sort后的二维数组。

代码如下:

irb(main):027:0> x
=> {5=>6, 1=>2, 3=>4}

irb(main):028:0> x.sort
=> [[1, 2], [3, 4], [5, 6]]

另外ruby有OrderedHash这个类,它可以让Hash元素按插入顺序来sort,类似于数组的能力。

7. 如何从Hash里添加和删除元素?

增加元素,直接定义Key/Value:

代码如下:

irb(main):040:0> x[7]=8
=> 8

irb(main):041:0> x
=> {5=>6, 1=>2, 7=>8, 3=>4}

删除元素,使用delete方法,参数是想删除的Key:

代码如下:

irb(main):042:0> x.delete(7)
=> 8

irb(main):043:0> x
=> {5=>6, 1=>2, 3=>4}

当然也可以用delete_if带条件的删除Hash元素。例如删除所有Key大于3的:

代码如下:

irb(main):074:0* x.delete_if do |k,v| k>3 end
=> {1=>2, 3=>4}

8. 如何查找Hash元素?

使用has_key?和has_value?来判断是否有对应键和值:

代码如下:

irb(main):052:0> x.has_key? 1
=> true

irb(main):053:0> x.has_value? 4
=> true

irb(main):054:0> x.has_key? 9
=> false

考虑到默认值的情况,不能直接用if hash[:key]来判断,例如:

代码如下:

irb(main):055:0> y=Hash.new(9)
=> {}

irb(main):056:0> y.has_key? :a
=> false

irb(main):057:0> puts 1 if y[:a]
1
=> nil

其他的查找方法,还有find、select等,例如找到所有Key大于2的元素:

代码如下:

irb(main):038:0> x.select do |k,v| k>2 end
=> [[5, 6], [3, 4]]

9. 可以反转Hash吗?

答案是可以,使用invert方法:

代码如下:

irb(main):011:0* x={1=>2,3=>4,5=>6}
=> {5=>6, 1=>2, 3=>4}

irb(main):012:0> x.invert
=> {6=>5, 2=>1, 4=>3}

10. Hash和数组如何转换?

Hash转换到数组很简单:

代码如下:

irb(main):020:0> x.to_a
=> [[5, 6], [1, 2], [3, 4]]

数组转换到Hash麻烦一点,自己写一个方法to_hash:

代码如下:

irb(main):023:0> class Array
irb(main):024:1>   def to_hash
irb(main):025:2>     unless size%2 == 0
irb(main):026:3>       raise "array with odd number of elements"
irb(main):027:3>     end
irb(main):028:2>     hash = Hash.new
irb(main):029:2>     0.step(size-1,2) {|x| hash[self[x]] = self[x+1] }
irb(main):030:2>     hash
irb(main):031:2>   end
irb(main):032:1> end
=> nil

irb(main):033:0> x=[1,2,3,4,5,6]
=> [1, 2, 3, 4, 5, 6]

irb(main):034:0> x.to_hash
=> {5=>6, 1=>2, 3=>4}

11. Hash的丢值行为?

任何时候修改了Hash的Key(记住Key是包括数组在内的目标),都会发生丢值行为。

代码如下:

irb(main):001:0> h=Hash.new
=> {}
irb(main):002:0> x=[1,2,3]
=> [1, 2, 3]
irb(main):003:0> h[x]=4
=> 4
irb(main):004:0> h[x]
=> 4
irb(main):005:0> x<<4
=> [1, 2, 3, 4]
irb(main):006:0> h[x]

上述修改了键x的值,那么这个键对应的Value就丢失了。
解决办法是rehash。任何时候修改了Key的值,都要rehash一次。

代码如下:

irb(main):007:0> h.rehash
=> {[1, 2, 3, 4]=>4}
irb(main):008:0> h[x]   
=> 4

(0)

相关推荐

  • Ruby中Hash哈希结构的基本操作方法小结

    关于哈希 先来了解一下Hash的基本思路: 设要存储对象的个数为num, 那么我们就用len个内存单元来存储它们(len>=num); 以每个对象ki的关键字为自变量,用一个函数h(ki)来映射出ki的内存地址,也就是ki的下标,将ki对象的元素内容全部存入这个地址中就行了.这个就是Hash的基本思路. 为什么要用一个函数来映射出它们的地址单元呢? 假设现在我要存储4个元素 13 7 14 11 显然,我们可以用数组来存.也就是:a[1] = 13; a[2] = 7; a[3] = 14; a

  • Ruby中的Hash哈希类型基本操作方法小结

    1.创建哈希:就像创建数组一样,我们可以通过Hash类来创建一个Hash实例: h1 = Hash.new #默认值为nil h2 = Hash.new("This is my first hash instance") #默认值为" This is my first hash instance": 上面两个例子都创建了一个空的Hash实例.一个Hash对象总是有一个默认的值--因为如果在一个Hash对象里没有找到指定的索引(key),将会返回默认值. 创建了Has

  • Ruby字符串、条件、循环、数组、Hash、类基本操作笔记

    一.字符串操作 字符串插值 1.#{}: 任意放置在#{}中的合法Ruby代码都将被求值,并被替换为求值结果插入到原位置 2.单引号与双引号差别: 双引号允许字符转义,单引号不允许转义,里是什么字符,用户看到的就是什么字符 单引号不允许字符插值 字符串拆分 1.以空格拆分:string.split(' ')字符串连接 方法一.'Ruby' + 'Monk', stdout: RubyMonk 方法二."Monk".concat("Ruby") stdout: Mon

  • Ruby简明教程之数组和Hash介绍

    今天学习的是数组和Hash,数组在很多编程语言中都能常常用到. 数组: 数组是存储数据的一种容器,在Ruby中,数组中存储的数据可以是任何类型的数据:这和JAVA不同,在JAVA中,数组是存储同一类型数据的结构. 1. 在Ruby中如何定义数组呢? 使用[ ]来框住元素,而元素之间则使用","间隔.如下, 复制代码 代码如下: name = ["Windy", "Cindy", "Amee", "Tom"]

  • Ruby中关于hash的基本使用方法

    哈希键 - 值对的集合,类似这样:"employee" => "salary".它类似于数组,除了索引是通过任意键的任何对象的类型,而不是一个整数索引. 顺序遍历一个哈希键或值要么可能看起来是任意的,一般不会在插入顺序.如果访问的键的哈希值不存在,该方法将返回nil. 创建哈希: 使用数组中,有各种各样的方法来创建哈希值.可以 new 类方法创建一个空的hash : months = Hash.new 还可以使用 new 创建哈希,这是一个默认值,否则只是 n

  • Ruby中Hash的11个问题解答

    最近在学习Rails,因为学习的java为入门语言.些许的不太适应.特别是Ruby的Hash.不过细细的想想就通了.基本上都是通的Key-Value.除了刚开始不太适应写法没什么大的问题.这个是我在网上看到的一篇日志. 1. 如何创建Hash? 复制代码 代码如下: x = Hash.new x = {} x = {:a => 1, :b => 2} 这些都是创建Hash的方法. 第一个和第二个一样,表示创建空的Hash. 第三个表示创建的Hash,有2对键/值. 2. Hash的键是哪些类型

  • Ruby中的钩子方法详解

    Ruby的哲学理念是基于一个基本的要素,那就是让程序员快乐.Ruby非常注重程序员的快乐,并且也提供了许多不同的方法来实现它. 它的元编程能力能够让程序员编写在运行时动态生成的代码.它的线程功能使得程序员有一种优雅的的方式编写多线程代码. 它的钩子方法能让程序员在程序运行时扩展它的行为. 上述的这些特性,以及一些其他很酷的语言方面,使得Ruby成为编写代码的优先选择之一. 本文将探讨Ruby中的一些重要的钩子方法.我们将从不同方面讨论钩子方法,如它们是什么,它们用于什么,以及我们如何使用它们来解

  • 详细解析Ruby中的变量

    变量持有要使用的程序的数据的存储位置. Ruby支持的有五种类型的变量.在前面的章节中已经经历了一个简短描述以及这些变量.本章中介绍的这五种类型的变量. Ruby的全局变量: 全局变量以$开头.未初始化的全局变量的值是零,并使用-w选项产生警告. 全局变量的赋值会改变全局状态.这是不推荐使用全局变量.他们使得程序的含义模糊. 下面是一个例子显示使用全局变量. #!/usr/bin/ruby $global_variable = 10 class Class1 def print_global p

  • Ruby中的迭代器详解

    D瓜哥最近想做一个网站,另外,老早就有学习一门动态语言的想法,满足着两个条件的编程语言中,Ruby.Python是最合适的两种语言.现在Ruby on Rails如日中天,光芒万丈!所以,就选定了Ruby,从零开始学习. 前天看了Ruby的迭代器,对于我这个只学过Java.C/C++等的人来说,绝对是眼前一亮的感觉!而且是光彩夺目:没想到迭代器还可以这么玩,太简练太方便而且特别强大!然后,D瓜哥就迫不及待的想写一篇文章给大家介绍介绍Ruby的迭代器! 迭代器简介 先简单介绍一下迭代器. 1.一个

  • 在Ruby中创建和使用哈希的教程

    哈希(Hash)是类似 "employee" => "salary" 这样的键值对的集合.哈希的索引是通过任何对象类型的任意键来完成的,而不是一个整数索引,其他与数组相似. 通过键或值遍历哈希的顺序看起来是随意的,且通常不是按照插入顺序.如果您尝试通过一个不存在的键访问哈希,则方法会返回 nil. 创建哈希 与数组一样,有各种不同的方式来创建哈希.您可以通过 new 类方法创建一个空的哈希: months = Hash.new 您也可以使用 new 创建带有默

  • Ruby中使用连续体Continuation实现生成器

    ruby中有很多经典的驱动器结构,比如枚举器和生成器等.这次简单介绍下生成器的概念.生成器是按照功能要求,一次产生一个对象,或称之为生成一个对象的方法.ruby中的连续体正好可以用来完成生成器的功能.连续体说起来晦涩,其实还是很简单的,它有3个特点: 1. callcc方法会给代码块传一个连续体对象,你可以保存该对象; 2. 当调用连续体的call方法时指令流会跳转到callcc方法之后; 3. 如果给连续体的call方法传递对象,则callcc方法会返回该对象,如果不传递对象,callcc会返

  • 深入理解Ruby中的代码块block特性

    block是什么? 在Ruby中,block并不罕见.官方对block的定义是"一段被包裹着的代码".当然,我觉得这样的解释不会让你变的更明白. 对block的一种更简单的描述是"一个block就是一段存储在一个变量中的代码,它和其他的对象一样,可以被随时的运行" 然后,咱们通过看一些代码,之后再把这些代码重构成Ruby中的block形式.通过代码来实际的感受,更加直观. 比如,对两个数做加法? puts 5 + 6 # => 11 嗯,这样写是可以的.但是,

  • Ruby中Time对象的常用函数总结

    时间对象. Time.now返回当前时间. 1.Time.at Time.at(time[, usec]) 返回time所指时间的Time对象. time可以是Time对象,也可以是表示自起算时间以来的秒数的整数或浮点数. 若浮点精度不够时,可以使用usec. 它将返回time +(usec/1000000)所表示的时间. 此时,time和usec都必需是整数. 生成的Time对象将使用地方时的时区. 2.Time.gm.Time.utc Time.gm(year[, mon[, day[, h

随机推荐