在前两篇文章中,为了运行 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 吧:
可以看出:
- 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 文件做相应修改,如下所示:
全部修改完成,开始生成目标程序:
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 进行汇编,结果汇编出错了。
今天就到此为止。以后如果能够找到解决方案再说吧。