javascript - Rubyracer(Ruby 的 V8 绑定(bind))执行速度非常慢

标签 javascript ruby v8 eventmachine embedded-v8

所以,我在 eventmachine 有一个 TCP 服务器和therubyracer用作对服务器预先挂起操作(如过滤器或扩展)的方式。当服务器没有接收大量数据时,这一切都很有吸引力,但是当它被淹没时(有时需要),它会变得非常慢。

因此,我做了一个小型基准测试,看看 ruby​​racer 与 Ruby 相比有多慢,当我看到结果时,我震惊:

          user     system      total        real
V8:     0.060000   0.000000   0.060000 (  0.059903)
Ruby:   0.000000   0.000000   0.000000 (  0.000524)

说实话,我不介意它是否慢,但我不希望它在完成数据处理之前锁定我的整个服务器。使用 EM::defer 并不是真正的选择(我尝试过,但有时它会产生大量线程,具体取决于洪水的强度)。我无法绕过洪水,因为我没有设计协议(protocol),并且客户端要求它们像那样(尽管它很可怕)。

基准代码:

require 'v8'
require 'benchmark'

class User
    def initialize
        @name = "smack"
        @sex = "female"
        @age = rand(100)
        @health = rand(100)
        @level = rand(100)
        @colour = rand(14)
    end
    
    attr_accessor :name, :sex, :age, :health, :level, :colour
end

# Create context and the function
context = V8::Context.new
code = "obj = {
    __incybincy__: function() {
        user.name + '' + '' + ''
        user.sex + '' + '' + ''
        user.age + '' + '' + ''
        user.health + '' + '' + ''
        user.level + '' + '' + ''
        user.colour + '' + '' + ''
    }
}"
context.eval(code)

# Insert the user into the context
user = User.new
context["user"] = user

# Benchmark
n = 100
Benchmark.bm do |x|
    x.report("V8: ") do 
        n.times do
            context['obj'].__incybincy__
        end
    end
    
    x.report("Ruby: ") do 
        n.times do
            user.name + "" + ""
            user.sex + "" + ""
            user.age.to_s + "" + ""
            user.health.to_s + "" + ""
            user.level.to_s + "" + ""
            user.colour.to_s + "" + ""
        end
    end
end

编辑

问题:有没有办法消除therubyracer造成的瓶颈?通过其他方式将 JavaScript 实现到 Ruby 中是可以接受的。


2012 年 3 月 7 日更新

因此,我设法优化了代码,因为我认为导致瓶颈的原因是 Ruby<->JS 通信,每次执行 [native code] 时都会发生这种情况,即自从 Ruby 对类使用 getter 和 setter 方法以来,或者当对象在语言之间直接传递时,一直如此。

                user     system      total        real
V8-optimized: 0.050000   0.000000   0.050000 (  0.049733)
V8-normal:    0.870000   0.050000   0.920000 (  0.885439)
Ruby:         0.010000   0.000000   0.010000 (  0.015064)
#where n is 1000

因此,我通过在 JS 端进行缓存来减少 Ruby 和 JS 之间的调用次数,但这并没有像我希望的那样优化它,因为至少需要将一个对象传递给函数: Hash 或至少是一个 JSON String,我什至不遗余力地传递一个 Fixnum——这让我感叹 FML——这不是比向其传递字符串(如果有的话)有很大的改进。

我仍然希望有一个比我的更好更快的解决方案。

最佳答案

问题是,默认情况下,Ruby Racer 字符串从 Ruby 复制到 V8,反之亦然。

在您的基准测试中,访问这 6 个字符串属性将导致至少 6 个 memcpy() 操作,这些操作必须分配新内存并逐字节遍历字符串的长度以移动它到新位置。将此与 Ruby 端进行比较,Ruby 端基本上是无操作的(字符串对象仅包装已分配和设置的指针),难怪它要慢得多。

您可以更改此行为以通过引用而不是通过值传递字符串。

class Wrapper
  attr_reader :object

  def inititialize(object)
    @object = object
  end
end

cxt['aString'] = Wrapper.new('not copied')

当然,如果你想在javascript中访问该字符串,你最终将不得不支付副本费用。您可以将这种包装技术用于数字、数组和哈希,默认情况下,所有这些都会复制到 JavaScript。

参见https://github.com/cowboyd/therubyracer/wiki/Converting-ruby-object-to-javascript了解更多详情。

V8 确实支持外部管理字符串的概念,这将允许您在 Ruby 中分配 char *,然后使用 V8 中的地址。然而,该功能目前在 The Ruby Racer 中不可用。

关于javascript - Rubyracer(Ruby 的 V8 绑定(bind))执行速度非常慢,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9573818/

相关文章:

页面加载时未调用 Javascript 代码? 。网

ruby - 查找最接近数组平均值的元素

javascript - 在 Django View 中传递列表或数组(作为参数)

ruby-on-rails - Django 还是 Ruby-On-Rails?

ruby - 从 ruby​​ 文件 + YARD 获取方法和注释

c++ - V8 多线程函数

javascript - BigQuery 支持哪些 JS 语法?

javascript - Promise Chaining 如何在内存中工作?

javascript - 滑出网站的着陆页

javascript - 如何在javascript中制作特殊格式的日期和时间