在前两篇文章中,为了运行 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 文件,在最后加入以下内容:
1 2 3 4 | # 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 :
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 | 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 程序中加入一句对这个类的调用,如下所示:
1 2 3 4 5 6 7 8 9 | 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 进行汇编,结果汇编出错了。
今天就到此为止。以后如果能够找到解决方案再说吧。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· [AI/GPT/综述] AI Agent的设计模式综述