- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
如果我有数组 a
和 b
,表达式 a-b
返回一个包含 a
中所有元素的数组> 不在 b
中。 “不在”在这里表示不等式 (!=
)。
在我的例子中,两个数组只包含相同类型的元素(或者,从 ducktyping 的角度来看,只包含理解“相等”方法 f
的元素)。有没有一种简单的方法可以将此 f
指定为相等的标准,以类似的方式我可以在执行 sort
时提供我自己的比较器?目前,我明确地实现了这一点:
# Get the difference a-b, based on 'f':
a.select { |ael| b.all? {|bel| ael.f != bel.f} }
这行得通,但我想知道是否有更简单的方法。
更新:从对这个问题的评论中,我得到的印象是,一个具体的例子将不胜感激。那么,我们开始吧:
class Dummy; end
# Create an Array of Dummy objects.
a = Array.new(99) { Dummy.new }
# Pick some of them at random
b = Array.new(10) { a.sample }
# Now I want to get those elements from a, which are not in b.
diff = a.select { |ael| b.all? {|bel| ael.object_id != bel.object_id} }
当然在这种情况下,我也可以说 ! ael eql? bel
,但在我的一般解决方案中,情况并非如此。
最佳答案
例如,“正常”对象相等性数组的哈希和集合操作(例如 -
操作)使用 Object#hash
的输出包含对象的方法以及 a.eql?(b)
比较的语义。
这可用于提高性能。 Ruby 在这里假设两个对象是 eql?
如果它们各自的 hash
方法的返回值相同(因此,假设两个对象返回不同的 hash
值不是 eql?
)。
对于正常的 a - b
操作,这可以用于首先计算每个对象的 hash
值一次,然后只比较这些值。这相当快。
现在,如果您有自定义相等性,最好的办法是覆盖对象的 hash
方法,以便它们返回适合这些语义的值。
一种常见的方法是构建一个包含所有数据的数组,其中包含对象标识的一部分并获取其哈希值,例如
class MyObject
#...
attr_accessor :foo, :bar
def hash
[self.class, foo, bar].hash
end
end
在您的对象的hash
方法中,您将包括您的f
比较方法当前考虑的所有数据。不是实际使用 f
,而是使用所有 Ruby 对象的默认语义,并且可以再次实现对对象的快速设置操作。
但是,如果这不可行(例如,因为您需要基于用例的不同等式语义),您可以自己模拟 ruby 的功能。
使用 f
方法,您可以按如下方式执行设置操作:
def f_difference(a, b)
a_map = a.each_with_object({}) do |a_el, hash|
hash[a_el.f] = a_el
end
b.each do |b_el|
a_map.delete b_el.f
end
a_map.values
end
使用这种方法,您只需计算每个对象的 f
值一次。我们首先用 a
中的所有 f
值和元素构建一个 HashMap ,并根据它们的 f< 从
值。其余值是结果。b
中删除匹配元素
这种方法使您不必为 a
中的每个对象遍历 b
,如果对象很多,这可能会很慢。但是,如果您的每个数组上只有几个对象,那么您原来的方法应该已经没问题了。
让我们看一下基准测试,我使用标准 hash
方法代替您的自定义 f
以获得可比较的结果。
require 'benchmark/ips'
def question_diff(a, b)
a.select { |ael| b.all? {|bel| ael.hash != bel.hash} }
end
def answer_diff(a, b)
a_map = a.each_with_object({}) do |a_el, hash|
hash[a_el.hash] = a_el
end
b.each do |b_el|
a_map.delete b_el.hash
end
a_map.values
end
A = Array.new(100) { rand(10_000) }
B = Array.new(10) { A.sample }
Benchmark.ips do |x|
x.report("question") { question_diff(A, B) }
x.report("answer") { answer_diff(A, B) }
x.compare!
end
使用 Ruby 2.7.1,我在我的机器上得到以下结果,显示问题的原始方法比我的答案的优化版本慢 5.9 倍:
Warming up --------------------------------------
question 1.304k i/100ms
answer 7.504k i/100ms
Calculating -------------------------------------
question 12.779k (± 2.0%) i/s - 63.896k in 5.002006s
answer 74.898k (± 3.3%) i/s - 375.200k in 5.015239s
Comparison:
answer: 74898.0 i/s
question: 12779.3 i/s - 5.86x (± 0.00) slower
关于ruby - 明确指定的方法或 block 的数组差异,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62656327/
我想了解 Ruby 方法 methods() 是如何工作的。 我尝试使用“ruby 方法”在 Google 上搜索,但这不是我需要的。 我也看过 ruby-doc.org,但我没有找到这种方法。
Test 方法 对指定的字符串执行一个正则表达式搜索,并返回一个 Boolean 值指示是否找到匹配的模式。 object.Test(string) 参数 object 必选项。总是一个
Replace 方法 替换在正则表达式查找中找到的文本。 object.Replace(string1, string2) 参数 object 必选项。总是一个 RegExp 对象的名称。
Raise 方法 生成运行时错误 object.Raise(number, source, description, helpfile, helpcontext) 参数 object 应为
Execute 方法 对指定的字符串执行正则表达式搜索。 object.Execute(string) 参数 object 必选项。总是一个 RegExp 对象的名称。 string
Clear 方法 清除 Err 对象的所有属性设置。 object.Clear object 应为 Err 对象的名称。 说明 在错误处理后,使用 Clear 显式地清除 Err 对象。此
CopyFile 方法 将一个或多个文件从某位置复制到另一位置。 object.CopyFile source, destination[, overwrite] 参数 object 必选
Copy 方法 将指定的文件或文件夹从某位置复制到另一位置。 object.Copy destination[, overwrite] 参数 object 必选项。应为 File 或 F
Close 方法 关闭打开的 TextStream 文件。 object.Close object 应为 TextStream 对象的名称。 说明 下面例子举例说明如何使用 Close 方
BuildPath 方法 向现有路径后添加名称。 object.BuildPath(path, name) 参数 object 必选项。应为 FileSystemObject 对象的名称
GetFolder 方法 返回与指定的路径中某文件夹相应的 Folder 对象。 object.GetFolder(folderspec) 参数 object 必选项。应为 FileSy
GetFileName 方法 返回指定路径(不是指定驱动器路径部分)的最后一个文件或文件夹。 object.GetFileName(pathspec) 参数 object 必选项。应为
GetFile 方法 返回与指定路径中某文件相应的 File 对象。 object.GetFile(filespec) 参数 object 必选项。应为 FileSystemObject
GetExtensionName 方法 返回字符串,该字符串包含路径最后一个组成部分的扩展名。 object.GetExtensionName(path) 参数 object 必选项。应
GetDriveName 方法 返回包含指定路径中驱动器名的字符串。 object.GetDriveName(path) 参数 object 必选项。应为 FileSystemObjec
GetDrive 方法 返回与指定的路径中驱动器相对应的 Drive 对象。 object.GetDrive drivespec 参数 object 必选项。应为 FileSystemO
GetBaseName 方法 返回字符串,其中包含文件的基本名 (不带扩展名), 或者提供的路径说明中的文件夹。 object.GetBaseName(path) 参数 object 必
GetAbsolutePathName 方法 从提供的指定路径中返回完整且含义明确的路径。 object.GetAbsolutePathName(pathspec) 参数 object
FolderExists 方法 如果指定的文件夹存在,则返回 True;否则返回 False。 object.FolderExists(folderspec) 参数 object 必选项
FileExists 方法 如果指定的文件存在返回 True;否则返回 False。 object.FileExists(filespec) 参数 object 必选项。应为 FileS
我是一名优秀的程序员,十分优秀!