请在2024年Clojure状态调查中分享您的想法!

欢迎!请查阅关于页面以获取更多关于如何使用本站的信息。

+1
ClojureCLR
    static void Main()
    {
        try
        {
            IFn load = clojure.clr.api.Clojure.var("clojure.core", "load");

            // Below code line seems to expect clojure code (non-compiled) in path
            // bin\Debug\net7.0\clojure\hello.test.clj
            // 
            // OR a compiled dll: 
            //
            // from folderstructure 
            // ./clojure/hello/test.clj
            // > clojure.compile clojure.hello.test
            // 
            // with content in clj file like:
            //   (ns clojure.hello.test)
            //   (defn output[] "ninja")
            var r = load.invoke("hello.test"); 
            IFn helloFn = clojure.clr.api.Clojure.var("clojure.hello.test", "output");
            var result = helloFn.invoke();
            Console.WriteLine("Calling compiled clj from C#, Got: " + result);
        }
        catch (Exception ex)
        {
            Console.WriteLine($"Error: {ex.Message}");
        }
    }
我的问题
有没有一个简单的示例来展示如何从C#代码中调用Clojure函数,就像上面那样?

有没有关于如何从C#与Clojure互操作时考虑命名空间和文件夹结构的一般文档说明?

我现在的主要目标是享受乐趣,并最终使用nREPL + CIDER成功运行一些东西。
我试图使用load.invoke加载(clojure.tools.nrepl),但在放置dll以及如何重命名它的地方感到困惑!

// 使用以下代码
IFn load = clojure.clr.api.Clojure.var("clojure.core", "load");
var r = load.invoke("tools.nrepl");

我输出了RT.cs静态FileInfo FindFile(string path, string filename)
probePath属性并发现以下内容
...是其框架文件夹的绝对路径(net7.0)

...\net7.0\clojure\tools.nrepl.cljr
...\net7.0\bin\clojure\tools.nrepl.cljr
...\net7.0\clojure\tools.nrepl.cljr
...\net7.0\clojure\tools.nrepl.cljc
...\net7.0\bin\clojure\tools.nrepl.cljc
...\net7.0\clojure\tools.nrepl.cljc
...\net7.0\clojure\tools.nrepl.clj
...\net7.0\bin\clojure\tools.nrepl.clj
...\net7.0\clojure\tools.nrepl.clj
...\net7.0\clojure.tools.nrepl.cljr.dll
...\net7.0\bin\clojure.tools.nrepl.cljr.dll
...\net7.0\clojure.tools.nrepl.cljr.dll
...\net7.0\clojure.tools.nrepl.cljc.dll
...\net7.0\bin\clojure.tools.nrepl.cljc.dll
...\net7.0\clojure.tools.nrepl.cljc.dll
...\net7.0\clojure.tools.nrepl.clj.dll
...\net7.0\bin\clojure.tools.nrepl.clj.dll
...\net7.0\clojure.tools.nrepl.clj.dll
错误:无法在加载路径上找到clojure/tools.nrepl中扩展名为.cljr, .cljc, .clj, .cljr.dll, .cljc.dll, 或 .clj.dll的文件。

这些都不匹配由nuget提供的clojure.tools.nrepl.dll
操作:重命名
clojure.tools.nrepl.dll -> clojure.tools.nrepl.clj.dll

然后dll被拾取,但加载失败
clojure.lang.Compiler.AssemblyInitializationException: '找不到初始化器
对于clojure.tools.nrepl, Version=0.1.0.0, Culture=neutral,
  PublicKeyToken=null.clojure/tools.nrepl'

已有 2 条回答

0 次投票

clojurians 上发布

0 次投票

我没有看到你调用Assembly.Load的地方,我认为这是必要的。请在 Program.cs 中尝试以下内容

using clojure.lang;
using System.Reflection;

// start up Clojure runtime including compiler
RT.Init();

// make .cljr file resources available from DLL
Assembly.Load("clojure.tools.nrepl");
Assembly.Load("clojure.tools.reader");

// compile nREPL lib
var ns = "clojure.tools.nrepl";
RT.var("clojure.core", "require").invoke(Symbol.intern(ns));

// create nREPL server (default port is 1667)
var startServer = RT.var(ns, "start-server!");
var server = startServer.invoke();

// allow exit on Ctrl-C by shutting down server
var stopServer = RT.var(ns, "stop-server!");
Console.CancelKeyPress += (_, _) => stopServer.invoke(server);

以下是合适的 example.csproj 文件

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <OutputType>Exe</OutputType>
    <TargetFramework>net7.0</TargetFramework>
    <ImplicitUsings>enable</ImplicitUsings>
    <Nullable>enable</Nullable>
  </PropertyGroup>

  <ItemGroup>
    <PackageReference Include="Clojure" Version="1.12.0-alpha8" />
    <PackageReference Include="clojure.tools.nrepl" Version="0.1.0-alpha1" />
  </ItemGroup>

</Project>

我在 Ubuntu 22.04 上使用 dotnet 7.0.114 。在空目录中创建这两个文件,只需运行

dotnet restore
dotnet run

祝您玩得开心!

谢谢您分享了这些,它们像魔术一样有效。也许也让我对 Clojure 的“内部”有一些理解:)  (RT.init, RT.var 等)。
我认为维基可以借鉴您上述的更多清晰的示例。
https://github.com/clojure/clojure-clr/wiki/Using-ClojureCLR-in-a-C%23-project

我会尝试学习更多,也许有一天会做出贡献...
...