为了训练自己一点 Typescript,我写了一个 simple ES6 Map+Set-like implementation based on plain JS Object .它只适用于原始键,所以没有桶,没有哈希码等。我遇到的问题是实现删除方法。使用普通的 delete 速度慢得令人无法接受。对于大型 map ,它比 ES6 map 删除慢 300-400 倍。我注意到如果对象的大小很大,性能会大幅下降。在 Node JS 7.9.0(例如 Chrome 57)上,如果对象具有 50855 个属性,delete 性能与 ES6 Map 相同。但是对于 50856 个属性,ES6 映射要快 2 个数量级。这是要重现的简单代码:
// for node 6: 76300
// for node 7: 50855
const N0 = 50855;
function fast() {
const N = N0
const o = {}
for ( let i = 0; i < N; i++ ) {
o[i] = i
}
const t1 = Date.now()
for ( let i = 0; i < N; i++ ) {
delete o[i]
}
const t2 = Date.now()
console.log( N / (t2 - t1) + ' KOP/S' )
}
function slow() {
const N = N0 + 1 // adding just 1
const o = {}
for ( let i = 0; i < N; i++ ) {
o[i] = i
}
const t1 = Date.now()
for ( let i = 0; i < N; i++ ) {
delete o[i]
}
const t2 = Date.now()
console.log( N / (t2 - t1) + ' KOP/S' )
}
fast()
slow()
我想我可以不使用 delete 属性,而是将它们设置为 undefined 或一些保护对象,但这会使代码困惑,因为 hasOwnProperty 将无法正常工作,for...in 循环将需要额外检查等。还有更好的解决方案吗?
附言我在 OSX Sierra 上使用节点 7.9.0
已编辑 感谢大家的评论,我修复了 OP/S => KOP/S。我想我问的问题不明确,所以我更改了标题。经过一些调查,我发现例如在 Firefox 中没有这样的问题——删除成本线性增长。所以这是 super 智能V8的问题。我认为这只是一个错误:(
最佳答案
(此处为 V8 开发人员。)是的,这是一个已知问题。潜在的问题是当对象变得太稀疏时,对象应该将它们的元素后备存储从平面数组切换到字典,并且历史上实现的方式是每个 delete 操作检查是否有足够的元素仍然存在,因为过渡还没有发生。数组越大,检查所花的时间就越多。在特定条件下(最近创建的对象小于特定大小),检查被跳过 - 由此产生的令人印象深刻的加速是您在 fast() 情况下观察到的。
我借此机会修复了常规/慢速路径的(坦率地说相当愚蠢的)行为。时不时地检查一下就足够了,而不是每次删除时都检查一下。修复将出现在 V8 6.0 中,Node 应该会在几个月内采用它(我相信 Node 8 应该会在某个时候得到它)。
也就是说,在许多情况下,使用 delete 会导致各种形式和幅度的减速,因为它会使事情变得更复杂,迫使引擎(任何引擎)执行更多检查和/或脱离各种快速路径。通常建议尽可能避免使用 delete。既然你有 ES6 map /集,就使用它们吧! :-)
关于javascript - V8中JS中对象属性的缓慢删除,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43594092/
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev
我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为
我希望我的UserPrice模型的属性在它们为空或不验证数值时默认为0。这些属性是tax_rate、shipping_cost和price。classCreateUserPrices8,:scale=>2t.decimal:tax_rate,:precision=>8,:scale=>2t.decimal:shipping_cost,:precision=>8,:scale=>2endendend起初,我将所有3列的:default=>0放在表格中,但我不想要这样,因为它已经填充了字段,我想使用占位符。这是我的UserPrice模型:classUserPrice回答before_val
查看Ruby的CSV库的文档,我非常确定这是可能且简单的。我只需要使用Ruby删除CSV文件的前三列,但我没有成功运行它。 最佳答案 csv_table=CSV.read(file_path_in,:headers=>true)csv_table.delete("header_name")csv_table.to_csv#=>ThenewCSVinstringformat检查CSV::Table文档:http://ruby-doc.org/stdlib-1.9.2/libdoc/csv/rdoc/CSV/Table.html
我有一个包含模块的模型。我想在模块中覆盖模型的访问器方法。例如:classBlah这显然行不通。有什么想法可以实现吗? 最佳答案 您的代码看起来是正确的。我们正在毫无困难地使用这个确切的模式。如果我没记错的话,Rails使用#method_missing作为属性setter,因此您的模块将优先,阻止ActiveRecord的setter。如果您正在使用ActiveSupport::Concern(参见thisblogpost),那么您的实例方法需要进入一个特殊的模块:classBlah
我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss
我正在尝试在我的centos服务器上安装therubyracer,但遇到了麻烦。$geminstalltherubyracerBuildingnativeextensions.Thiscouldtakeawhile...ERROR:Errorinstallingtherubyracer:ERROR:Failedtobuildgemnativeextension./usr/local/rvm/rubies/ruby-1.9.3-p125/bin/rubyextconf.rbcheckingformain()in-lpthread...yescheckingforv8.h...no***e
好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信
我有一个具有一些属性的模型:attr1、attr2和attr3。我需要在不执行回调和验证的情况下更新此属性。我找到了update_column方法,但我想同时更新三个属性。我需要这样的东西:update_columns({attr1:val1,attr2:val2,attr3:val3})代替update_column(attr1,val1)update_column(attr2,val2)update_column(attr3,val3) 最佳答案 您可以使用update_columns(attr1:val1,attr2:val2