单声道编译器即服务(MCS)

时间:2010-08-04 15:49:09

标签: c# .net compiler-construction mono gmcs

我想将Mono的编译器作为常规.NET 3.5应用程序中的服务使用。

我已经下载了最新的位(2.6.7),在Visual Studio中创建了一个简单的控制台应用程序并引用了Mono.CSharp dll。

然后,在我的控制台应用程序中(直接在线样本):

    Evaluator.Run("using System; using System.Linq;");
    bool ress;
    object res;
    Evaluator.Evaluate(
         "from x in System.IO.Directory.GetFiles (\"C:\\\") select x;",
         out res, out ress);

    foreach (var v in (IEnumerable)res)
    {
        Console.Write(v);
        Console.Write(' ');
    }

这会在Evaluator.Run(第一行)抛出异常:

Illegal enum value: 2049.
Parameter name: access

这是因为dll是使用Mono.exe编译的,而不是csc.exe,我相信。

我尝试直接从demo-repl.zip文件中的http://tirania.org/blog/archive/2010/Apr-27.html下载Mono.CSharp dll ...并且不会抛出异常...但是调用后输出out参数(res) Evaluator.Evaluate是null ...所以我不确定出了什么问题。没有例外被抛出......

所以,我想弄清楚为什么我从demo-repl.zip下载的dll返回null。

编辑:我弄清楚它为什么返回null。看起来由于某种原因,编译器没有拿起System.Linq命名空间......虽然我不知道为什么......如果我只是评估“System.IO.Directory.GetFiles(\”C:\\ “)”,它运作正常。

更新:Mono编译器选择引用的系统程序集似乎有些不对劲。如果我直接复制其csharp控制台工具的样本:

csharp> var list = new int [] {1,2,3};
csharp> var b = from x in list
   >    where x > 1
   >    select x;
csharp> b;

我得到例外:

{interactive}(1,25): error CS1935: An implementation of `Select' query expressio
n pattern could not be found. Are you missing `System.Linq' using directive or `
System.Core.dll' assembly reference?

另外,为了让MCS实际上成为一个可行的解决方案,我需要修改编译器,使其发送到一个动态程序集,而不是每次评估调用发出一个程序集(否则它会呈现一个主要内存)泄漏,我之前以CSharpCodeProvider的形式处理过。有没有人知道这会有多困难,或者有人能指出我在正确的方向吗?

感谢。

2 个答案:

答案 0 :(得分:2)

好的,我想我有一些答案。

要解决程序集加载问题,我可以在Mono.CSharp.Driver.LoadAssembly中调用Assembly.LoadWithPartialName,或者在我的应用程序中执行以下操作

        AppDomain.CurrentDomain.AssemblyResolve += new ResolveEventHandler(CurrentDomain_AssemblyResolve);

        private static bool isResolving;
        static System.Reflection.Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
        {
            if (!isResolving)
            {
                isResolving = true;
                var a = Assembly.LoadWithPartialName(args.Name);
                isResolving = false;
                return a;
            }
            return null;
        }

为了让Mono为每个Evaluate / Compile调用重用相同的动态程序集,我必须改变的是以下内容(虽然我可能在这里找不到复杂的内容).....

在Mono.CSharp.Evaluator中,我添加了属性:

/// <summary>
/// Gets or sets a value indicating whether to auto reset when evaluations are performed and create a new assembly.
/// </summary>
/// <value><c>true</c> if [auto reset]; otherwise, <c>false</c>.</value>
public static bool AutoReset { get; set; }

然后......确保在Init:

中至少调用一次Reset
    static void Init ()
    {
        Init (new string [0]);
        Reset();
    }

最后,在ParseString中,除非AutoReset为真,否则不要重置...

        static CSharpParser ParseString (ParseMode mode, string input, out bool partial_input)
        {
.
.
.
            if (AutoReset) Reset ();

答案 1 :(得分:1)

根据您链接的Miguel博客页面,您必须添加对System.Core的引用才能在.Net上使用LINQ。

csharp> using System.Linq;
csharp> from x in "Foo" select x;