问题 使用JRuby / Jython实现Ruby / Python互操作性?


很可能是一个愚蠢的问题,因为我对Java / Jython / JRuby /字节码不太了解,但..

我偶然发现了 _为什么不圣洁 今天再次..它允许您从Ruby代码输出Python字节码。基本上允许它们生成相同的字节码..

Jython输出Java字节码,就像JRuby一样。由于这些都编译为相同的字节码,这是否意味着您可以使用Ruby中的任何Python库和Python中的Ruby库?


1335
2018-05-02 09:25


起源

+1有趣的问题...... - ChristopheD
顺便说一句:那个“运行所有语言的神秘虚拟机”,这是在邪恶的自述文件的最后提到的, 是 我在答案中提到的Parrot VM。值得庆幸的是,它现在已不再是神话般的了:大型1.0.0版本是在一年多前发布的,从那时起,Parrot团队每月发布一个稳定版本,如发条,两周前到达(目前)2.3.0 。当然,这并不意味着确实存在 编译器 对于每种语言(特别是Cardinal Ruby编译器远非生产就绪),只有VM存在且稳定。 - Jörg W Mittag


答案:


不,那不行。至少不是你想象的方式。

Jython和JRuby之间的互操作性与CPython和YARV之间的互操作方式相同:它们都在同一平台上运行,因此它们可以使用该平台相互通信。

在CPython和YARV的情况下,该平台是C / POSIX,因此它们可以使用C结构相互通信, intS, char*s和C函数调用。对于Jython和JRuby,该平台是JVM,因此它们可以使用JVM对象,JVM类,JVM接口,JVM类型和JVM方法相互通信。

在这两种情况下,这些平台基元看起来都是 没有 像Python或Ruby对象。

对JRuby来说,Jython只是另一个Java程序。对于Jython,JRuby只是另一个Java程序。

例如:在Ruby中,您可以随时动态添加,删除和重新定义方法。在JVM上,可以动态添加和删除的最小代码单元是一个类。因此,Ruby方法实际上并未表示为Java方法。它表示为Java 。从逻辑上讲,一个带有几个方法的Ruby对象被表示为没有方法的Java对象,只有一个 Dictionary<String, RubyMethod> 领域。 IOW:它完全不能用于Java,而且,从JRuby的角度来看,Jython只是Java,它也无法从Jython中使用。

现在,那里  如何使这更好一点。您可以使用实际的Java类型在两者之间进行通信 - 两种实现都与Java具有很好的互操作性。因此,不是将Ruby哈希传递给Python或Python字典传递给Ruby,而是使用Java Map 来自Ruby和Python。但请注意,这需要  您的Ruby和Python代码专门用于在JVM上运行。 IOW:你不能只使用你在网上找到的任何Python或Ruby库,这就是你所要求的。

另一种可能性是@duncan在他的回答中提到的:将Jython或JRuby作为脚本引擎嵌入到Ruby或Python应用程序中。但同样,这并没有真正回答你关于使用Ruby的任意Python库的问题,反之亦然。

那么,这里的问题是什么?

问题是,为了使两个运行时进行通信,它们需要说出相同的“语言”。在这种特殊情况下,两个运行时共有的唯一语言是Java,或者更确切地说是Java的严重残缺子集。

所以,我们需要找到一种共同的语言。定义这种语言的一种方法是两个运行时都能理解彼此的元对象协议(MOP)。

MOP基本上是语言对象模型的对象模型。嗯,这令人困惑,因为我们使用“对象模型”这个词来表示两种不同的东西。让我重新说一下:

MOP基本上是语言对象系统的域模型。就像银行系统的域模型包含代表真实客户,账户,余额,分类账等的对象,以及代表货币转移,提款等真实行为的方法,MOP包含代表的对象语言类,方法,变量,对象和方法,表示语言操作,如查找变量,调用方法,继承类,构造类的实例。

通常,每个运行时都将其MOP保持为私有,并且每个运行时都有自己的MOP。

如果JRuby和Jython将他们的MOP暴露给对方并理解彼此的MOP(或者更好的是:他们将他们的MOP暴露给JVM并且都使用了 一样 MOP),然后你可以将其中一个疯狂的JRuby方法包传递给Jython,它会知道如何找到属于该对象的方法以及如何调用它们,因为它可以只询问JRuby的MOP如何做到这一点。

实际上有一个项目可以为JVM创建这样一个MOP: dynalang MOP 是一个用于在JVM上运行的动态语言的共享标准化MOP的项目。它是由Mozilla Rhino ECMAScript引擎的维护者Attila Szegedi创建的。目前,大型语言实现都没有使用它,但至少在Rhino,JRuby,Jython和Groovy之间进行协作,以确保dynalang足够通用,可以支持所有不同语言的对象模型。

如果你想看看这样一个共享MOP的世界会是什么样子,你可以看看微软的动态语言运行时(DLR)。 DLR只包含这样一个MOP和支持DLR的所有运行时(除了通常的嫌疑人之外) IronRuby的IronPython的IronJS 和 IronScheme 现在还包括C#4和Visual Basic.NET 10)几乎可以互相无缝地互操作。

另一个类似的平台是 鹦鹉虚拟机,专门设计用于允许多个动态语言在同一运行时平台上进行互操作。有Python的实现(Pynie)和Ruby(枢机主教)可用,但特别是Cardinal仍然远远不是一个远程完整的Ruby实现。


7
2018-05-02 15:58





有两种方法可以做到这一点。两者都提供了静态编译代码并从脚本生成真正的Java类的能力。在这种情况下,Jython AFAIK生成Java源代码,然后通过jythonc脚本调用javac。但这需要编译。

对于这两个解释器,您可以从脚本调用Java代码,并且可以将解释器嵌入Java应用程序中。

例如,从Python调用Java:

>>> from java.util import Random
>>> r = Random()
>>> r.nextInt()
501203849

要在Java中嵌入JRuby解释器,你可以这样做(注意,也有基于JSR223的方法,这是核心方式):

package vanilla;

import org.jruby.embed.ScriptingContainer;

public class HelloWorld {

    private HelloWorld() {
        ScriptingContainer container = new ScriptingContainer();
        container.runScriptlet("puts Hello world");
    }

    public static void main(String[] args) {
        new HelloWorld();
    }

你可以从Jyton做同样的事情(我想你需要正确地给出jruby路径):

import org.jruby.embed.ScriptingContainer
container = ScriptingContainer()
container.runScriptlet("puts Hello world")

反过来也可以这样做。

你不会通过导入将整个ruby stdlib导出到python解释器。您需要事先将ruby的stdlib预编译为字节码。

但是,通过上述技术,并添加了几个辅助脚本和已定义的接口,您可以将特定功能从一种语言桥接到另一种语言。


5
2018-05-02 14:51