也谈在 .NET 平台上使用 Scala 语言(下)

在前两篇文章中,为了运行 Scala.NET 程序,我们要将 predef.dll 拷贝到当前目录下。这很不爽。 :(

我首先想到的是将 predef.dll 加入到全局程序集缓存(GAC, Global Assembly Cache)中去,如下所示:

ben@ben-m4000t:~$ sudo gacutil -i /opt/scala-2.7.7.final/lib/predef.dll
Failure adding assembly /opt/scala-2.7.7.final/lib/predef.dll to the cache: Attempt to install an assembly without a strong name

非常遗憾,predef.dll 不是强名(strong name)的,无法放入到全局程序集缓存中。这里我要批评 scala-msil 软件包的作者一下,这么重要的东东怎么能不是强名滴?

 

那么,我就另外想办法。首先,在 /usr/local/lib 目录下建立一个 mono-extra 的目录,在该目录下建立所需要的 predef.dll 和 scalaruntime.dll 的符号连接,如下所示:

ben@ben-m4000t:~$ cd /usr/local/lib
ben@ben-m4000t:/usr/local/lib$ sudo mkdir mono-extra
ben@ben-m4000t:/usr/local/lib$ cd mono-extra
ben@ben-m4000t:/usr/local/lib/mono-extra$ sudo ln -s /opt/scala-2.7.7.final/lib/predef.dll
ben@ben-m4000t:/usr/local/lib/mono-extra$ sudo ln -s /opt/scala-2.7.7.final/lib/scalaruntime.dll
ben@ben-m4000t:/usr/local/lib/mono-extra$ ls -l *
lrwxrwxrwx 1 root root 37 2009-12-23 14:36 predef.dll -> /opt/scala-2.7.7.final/lib/predef.dll
lrwxrwxrwx 1 root root 43 2009-12-23 14:37 scalaruntime.dll -> /opt/scala-2.7.7.final/lib/scalaruntime.dll
ben@ben-m4000t:/usr/local/lib/mono-extra$

然后,编辑 ~/.bashrc 文件,在最后加入以下内容:

# set mono extra library path
if [ -d "/usr/local/lib/mono-extra" ] ; then
    export MONO_PATH=/usr/local/lib/mono-extra
fi

这样,就可以不需要拷贝 predef.dll 到当前目录而正常运行 Scala.NET 程序了。

也可以把你自己想要共享的 dll 文件进行同样的处理。

注意,不要把 /opt/scala-2.7.7.final/lib/mscorlib.dll 也符号连接到 /usr/local/lib/mono-extra 目录下。否则,运行所有的 .NET 程序都会出现以下错误:

ben@ben-m4000t:~/Projects/ScalaNet$ mono dotnet.exe
Corlib not in sync with this runtime: expected corlib version 69, found 65.
Loaded from: /opt/scala-2.7.7.final/lib/mscorlib.dll
Download a newer corlib or a newer runtime at http://www.go-mono.com/daily.

这是因为 Scala.NET 的 mscorlib.dll 版本 corlib version 65 较旧,而 CLR 期望比较新的版本 corlib version 69。

 

那么,我们来看看这台机上各个 mscorlib.dll 文件的版本情况吧。

2,070,528 2009-12-22 22:52 /opt/scala-2.7.7.final/lib/mscorlib.dll
2,076,672 2009-09-23 23:28          /usr/lib/mono/1.0/mscorlib.dll
2,092,544 2009-12-23 20:09 /opt/mono-2.6/lib/mono/1.0/mscorlib.dll
2,565,632 2009-09-23 23:29          /usr/lib/mono/2.0/mscorlib.dll
2,586,624 2009-12-23 20:11 /opt/mono-2.6/lib/mono/2.0/mscorlib.dll

通过使用 monodis 工具反汇编的结果,我们得知以下信息(GUID 唯一标识了该文件):

.module mscorlib.dll // GUID = {F9EA90BC-5E97-4A29-A0A4-7F777CF93BC8} // Scala.NET
.module mscorlib.dll // GUID = {95A996E8-A93F-4388-A474-45EEDA546579} // mono 2.4.2.3 v1.0
.module mscorlib.dll // GUID = {9D069D60-9C3E-4912-90E1-B962D1A3C669} // mono 2.6.1 v1.0
.module mscorlib.dll // GUID = {32F1F939-70A3-452B-A8B7-FB4AB7284ACB} // mono 2.4.2.3 v2.0
.module mscorlib.dll // GUID = {7B63199A-BD44-405E-8C1F-EE5E2B4F2CA6} // mono 2.6.1 v2.0.0

这五个 mscorlib.dll 可以分为如下的两组(前三个属于一组,后两个属于另外一组):

mscorlib, Version=1.0.5000.0, Culture=neutral, PublicKeyToken=b77a5c561934e089
mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089

这两组的详细的情况如下表所示:

Item MonoVersion .ver FxVersion FxFileVersion VsVersion VsFileVersion
Scala.NET 1.9.1.0 1:0:5000:0 1.0.5000.0 1.1.4322.2032 7.0.5000.0 7.10.6001.4
2.4.2.3 v1.0 2.4.2.3 1:0:5000:0 1.0.5000.0 1.1.4322.2032 7.0.5000.0 7.10.6001.4
2.6.1 v1.0 2.6.1 1:0:5000:0 1.0.5000.0 1.1.4322.2032 7.0.5000.0 7.10.6001.4
2.4.2.3 v2.0 2.4.2.3 2:0:0:0 2.0.0.0 2.0.50727.1433 8.0.0.0 8.0.50727.1433
2.6.1 v2.0 2.6.1 2:0:0:0 2.0.0.0 2.0.50727.1433 8.0.0.0 8.0.50727.1433

 

接着,我们来看看用 monodis 反汇编出来的 predef.il 和 scalaruntime.il 吧:

也谈在 .NET 平台上使用 Scala 语言(下)_第1张图片

也谈在 .NET 平台上使用 Scala 语言(下)_第2张图片

 

可以看出:

  • predef 依赖 mscorlib 1:0:5000:0 以及 scalaruntime 0:0:0:0
  • scalaruntime 依赖 mscorlib 2:0:0:0

正如老赵所说,这够混乱的。

 

我们现在来编写一个复杂点的 C# 程序 TheXmlTree.cs :

using System;
using System.Linq;
using System.Xml.Linq;

namespace Skyiv
{
  public static class TheXmlTree
  {
    public static XElement GetValue()
    {
      var v = new XElement("江湖人物榜",
        new XElement("田伯光", 3),
        new XElement("令狐冲", 1),
        new XElement("岳不群", 4),
        new XElement("仪琳", 2),
        new XElement("林平之", 6),
        new XElement("任盈盈", 5)
      );
      return new XElement("笑傲江湖",
        from el in v.Elements()
        where (int)el >= 2 && (int)el <= 5
        select el
      );
    }
  }
}

然后在 dotnet.scala 程序中加入一句对这个类的调用,如下所示:

import System.Console
object dotnet extends Application {
  Console.WriteLine("       Scala.NET: 欢迎光临");
  Console.WriteLine("      OS Version: " + Environment.OSVersion);
  Console.WriteLine("     CLR Version: {0}  ( {1} )", Environment.Version, Skyiv.RuntimeFramework.CurrentFramework);
  Console.WriteLine("Default Encoding: " + System.Text.Encoding.Default);
  Console.WriteLine();
  Console.WriteLine(Skyiv.TheXmlTree.GetValue());
}

最后,对 makefile 文件做相应修改,如下所示:

也谈在 .NET 平台上使用 Scala 语言(下)_第3张图片

 

全部修改完成,开始生成目标程序:

ben@ben-vbox:~/Projects/ScalaNet$ make
csc -out:RuntimeFramework.dll -t:library RuntimeFramework.cs
csc -out:TheXmlTree.dll -t:library -r:System.Xml.Linq.dll TheXmlTree.cs
/opt/scala-2.7.7.final/bin/scalac-net -Xassem-path RuntimeFramework.dll:TheXmlTree.dll dotnet.scala
8@(06 15 12 1d 02 12 11 02)
error: error while loading TheXmlTree, type 'Skyiv.TheXmlTree' is broken
(15@2 in (06 15 12 1d 02 12 11 02))
dotnet.scala:9: error: value GetValue is not a member of object Skyiv.TheXmlTree
  Console.WriteLine(Skyiv.TheXmlTree.GetValue());
                                     ^
two errors found
make: *** [dotnet.msil] 错误 1
ben@ben-vbox:~/Projects/ScalaNet$

非常遗憾,在使用 scalac-net 将 dotnet.scala 源程序转换为 dotnet.msil 微软中间语言这一步出错了。

出错的地方是: type 'Skyiv.TheXmlTree' is broken

这应该是由于 predef.dll 依赖低版本的 mscorlib 1:0:5000:0 引起的。

最好的解决方案是 scala-msil 的作者改用 mscorlib 2:0:0:0 。

我试图修改 predef.il ,将其中 mscorlib 的 .ver 1:0:5000:0 改为 .ver 2:0:0:0,然后再用 ilasm 进行汇编,结果汇编出错了。

今天就到此为止。以后如果能够找到解决方案再说吧。

你可能感兴趣的:(scala)