在前两篇文章中,为了运行 Scala.NET 程序,我们要将 predef.dll 拷贝到当前目录下。这很不爽。 :(
我首先想到的是将 predef.dll 加入到全局程序集缓存(GAC, Global Assembly Cache)中去,如下所示:
非常遗憾,predef.dll 不是强名(strong name)的,无法放入到全局程序集缓存中。这里我要批评 scala-msil 软件包的作者一下,这么重要的东东怎么能不是强名滴?
那么,我就另外想办法。首先,在 /usr/local/lib 目录下建立一个 mono-extra 的目录,在该目录下建立所需要的 predef.dll 和 scalaruntime.dll 的符号连接,如下所示:
然后,编辑 ~/.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 程序都会出现以下错误:
这是因为 Scala.NET 的 mscorlib.dll 版本 corlib version 65 较旧,而 CLR 期望比较新的版本 corlib version 69。
那么,我们来看看这台机上各个 mscorlib.dll 文件的版本情况吧。
通过使用 monodis 工具反汇编的结果,我们得知以下信息(GUID 唯一标识了该文件):
这五个 mscorlib.dll 可以分为如下的两组(前三个属于一组,后两个属于另外一组):
这两组的详细的情况如下表所示:
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 吧:
可以看出:
- 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 文件做相应修改,如下所示:
全部修改完成,开始生成目标程序:
非常遗憾,在使用 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 进行汇编,结果汇编出错了。
今天就到此为止。以后如果能够找到解决方案再说吧。