.NET DLR 上的IronScheme 语言互操作&&IronScheme控制台输入中文的问题

简介:

前言

一直以来对Lisp语言怀有很崇敬的心里,《黑客与画家》对Lisp更是推崇备至,虽然看了不少有关Lisp的介绍但都没有机会去写段程序试试,就像我对C++一样,多少有点敬畏。这个周末花了不少时间来研究Lisp。
Lisp是古老的函数式语言,跟C,C++等命令式语言完全不一样的编程风格,但Lisp的方言很多,最后Lisp标准委员制定了Common Lisp,但内容很长,有1000多页,因此功能比较强大;而Lisp的另外一个主要分支就是Scheme,它的标准内容只有不到100页,所以非常简单,适合学术研究和大学计算机语言教学以及一般的工程应用。目前Lisp有在JVM上的实现,在.NET上的实现就是 IronScheme,于是我便开始选择了IronScheme作为Lisp研究的第一站。

1,下载IronScheme源码

IronScheme在Codeplex上有开源项目, https://ironscheme.codeplex.com/ ,可以下载它的源码和编译好的程序,在 https://ironscheme.codeplex.com/SourceControl/latest 可以下载源码,我下载时候的文件名是 ironscheme-103684,下载的源码可以用VS2008打开。如果没有开发环境,直接用 debugbuild.bat 也就可以直接编译。另外还可以直接运行测试 r6rstest.bat

2,IronScheme控制台

在网站上下载IronScheme的应用程序后,可以直接看到它已经提供了不同环境下的控制台程序,分别有64位与32位,.NET 2.0与4.0的程序: IronScheme.Console32-v2.exe IronScheme.Console32-v4.exe IronScheme.Console-v2.exe IronScheme.Console-v4.exe

2.1,执行Scheme程序

找一个合适的控制台运行下,输入几个Lisp表达式看看:

Lisp程序有一个天然的执行多个参数运算的特点,所以我们可以执行多个数字相加。也可以使用 display 函数显示一个字符串。

2.2,中文乱码问题

写一个简单的Hello 程序文件来加载 试试:

 执行这个程序,成功 ,但是乱码,不管是存储成 ANSI格式还是UTF8格式均乱码:

2.3,解决乱码

无奈,只有打开IronScheme源码进行分析,分析了很久很久....

最后干脆直接搜索编码格式 Encoding...,好歹涉及这个关键词的地方只有3个:

在 IronScheme.Console 项目下的 Program 文件中,找到下面的代码:

复制代码
 Encoding oo = Console.OutputEncoding;

      EnableMulticoreJIT();

      try
      {
          //Console.OutputEncoding = Encoding.UTF8;
        return IronSchemeConsoleHost.Execute(args);
      }
      finally
      {
        Console.OutputEncoding = oo;
      }
复制代码

将原来的 Console.OutputEncoding = Encoding.UTF8 注释即可,由于我的电脑是中文环境,这样程序便以GBK的编码运行了,此时即可正常显示Scheme 程序中的 汉字。但是,如果要加载的文件名有汉字,则悲剧了,控制台无法输入汉字...

再次检查程序中所有跟控制台有关的编码的地方,发现除了前面检查过的编码问题,再也没有其它地方,最后跟踪调试代码,发现程序使用

Console.ReadKey()

方法来获取屏幕输入的,而这个方法,是无法获得中文输入的...%&*....

 既然是截获了键盘敲击,那么我就顶一个特殊的键,按下它在弹窗出来一个窗口,在里面输入中文就可以了吧,于是找到文件 SuperConsole.cs ,找到 Insert(ConsoleKeyInfo key) 方法,修改成下面的代码:

复制代码
 private void Insert(ConsoleKeyInfo key) {
            char c;
            if (key.Key == ConsoleKey.F6)
            {
              Debug.Assert(FinalLineText.Length == 1);
              
              c = FinalLineText[0];
            }
            else if (key.Modifiers == ConsoleModifiers.Alt && (key.Key >= ConsoleKey.NumPad0 && key.Key <= ConsoleKey.NumPad9))
            {
              c = '?';
            }
            else
            {
              c = key.KeyChar;
            }
            //Ctrl+Z 输入汉字
            if (key.Key == ConsoleKey.Z && key.Modifiers == ConsoleModifiers.Control)
            {
                frmInputString frm = new frmInputString();
                frm.Activate();
                frm.ShowDialog();
                //Console.Write(frm.Text);
                string s = frm.Text;
                
                _input.Append(s);
                Output.Write(s);
                _rendered += s.Length;
                _current += s.Length;
            }
            else
            {
                Insert(c);
            }
        }
复制代码

这样就可以在Scheme控制台弹窗输入中文了,顺便加入文件选择功能,方便加载程序文件,如图:

控制台默认的字体是 “点阵字体”,这种字体在输入中文后,Scheme 定位字符位置会有问题,应该使用非点阵字体,例如如下图的设置(控制台窗口标题--属性--字体):

3,Scheme 调用 .NET

按照 作者官方的说法,IronScheme是可以签入在.NET应用程序里面的,但是单独执行Scheme程序的时候,是否可以调用 .net已有的程序呢?这个IronScheme也提供了,下面是 https://ironscheme.codeplex.com/wikipage?title=clr-syntax&referringTitle=Documentation 页面的内容:

These macro's are exported from the (ironscheme clr) library.

Common parameters

type is either:

  • a symbol. Eg: Int32 or System.IO.Stream
  • a list implying a generic type. Eg: (Action Int32)
  • #f (false) meaning the type should try to be inferred

Primary syntax

(clr-namespaces) Returns all the imported at the lexical scope
(clr-reference referencereference can be a symbol for the assembly short name (ie System.Web) or a string containing the fully qualified assembly name.
(clr-using namespacenamespace is a symbol. Eg System.IO .
(clr-call type method instance arg ...method is a symbol for a simple name, eg ToInt32 or a string to resolve specific methods, eg "ToInt32(Object)" . instance is a reference to the object of type . Can be null ('()) for static methods. arg ... is the arguments passed to the method.
(clr-cast type exprexpr is the instance to be cast.
(clr-is type exprexpr is the instance to be tested.
(clr-new type arg ...arg ... is the arguments passed to the constructor.
(clr-new-array type sizesize is the size of the array. Must be an integer.
(clr-event-add! type  event instance handlerevent is a symbol for the name of the event. Eg Click . instance is a reference to the object of type . Can be null ('()) for static events. handler is a procedure taking the same number of arguments as the event's delegate.
(clr-event-remove! type  event instance handlerevent is a symbol for the name of the event. Eg Click . instance is a reference to the object of type . Can be null ('()) for static events. handler is a procedure taking the same number of arguments as the event's delegate.
(clr-field-get type field instancefield is a symbol for the name of the field. Eg m_foo . instance is a reference to the object of type . Can be null ('()) for static fields.
(clr-field-set! type field instance exprfield is a symbol for the name of the field. Eg m_foo . instance is a reference to the object of type . Can be null ('()) for static fields. expr is the value to set the field.
(pinvoke-call library method arg ...arg ... is the arguments passed to the method.

Derived syntax

(clr-indexer-get type instance arg ...instance is a reference to the object of type . arg ... is the arguments passed to the indexer.
(clr-indexer-set! type instance arg ... exprinstance is a reference to the object of type . arg ... is the arguments passed to the indexer. expr is the value to set the indexer.
(clr-prop-get type property instanceproperty is the name of the property. Eg Height . instance is a reference to the object of type . Can be null ('()) for static properties.
(clr-prop-set! type property instance exprproperty is the name of the property. Eg Height . instance is a reference to the object of type . Can be null ('()) for static properties. expr is the value to set the property.
(clr-static-call type method arg ...method is a symbol for a simple name, eg ToInt32 or a string to resolve specific methods, eg "ToInt32(Object)" . arg ... is the arguments passed to the method.
(clr-static-event-add! type  event handlerevent is a symbol for the name of the event. Eg Click . handler is a procedure taking the same number of arguments as the event's delegate.
(clr-static-event-remove! type  event handlerevent is a symbol for the name of the event. Eg Click . handler is a procedure taking the same number of arguments as the event's delegate.
(clr-static-field-get type fieldfield is a symbol for the name of the field. Eg m_foo .
(clr-static-field-set! type field exprfield is a symbol for the name of the field. Eg m_foo . expr is the value to set the field.
(clr-static-prop-get type propertyproperty is the name of the property. Eg Height .
(clr-static-prop-set! type property exprproperty is the name of the property. Eg Height . expr is the value to set the property.

 

3.1,小试牛刀

看来支持得还挺全面,马上写个程序试试看:

复制代码
(import
  (rnrs)
  (ironscheme clr))

;Define a function write-ln
(define (write-ln fmt . args)
  (clr-static-call System.Console WriteLine (clr-cast System.String fmt)
    (clr-cast System.Object[]
      (list->vector args))))

; And invoke it!
(write-ln "{0}" "Hello World!")
(write-ln "1:{0}" "aaa")
(write-ln "1:{0} 2:{1}" "张三" "李四")
复制代码

这个程序是调用 .net的 Console.WriteLine 方法的,运行这个程序试试:

注意程序文件需要保存为 UTF8格式的,IronScheme 才可以正常显示中文。

3.2,为是么要用 Scheme调用 .NET?

利用 Lisp的强大表达能力,调用.net强大的类库

Scheme可以当作脚本语言,可以.net程序动态生成一个 Scheme程序,Scheme程序再调用.net。。。。 这个过程的用途,明白了吧?

比如工作流程序,调用一个Scheme 脚本..

 

参考资源

更多的 Lisp,Scheme学习资源,可以参考下面的链接 :

Lisp 的永恒之道
http://www.oschina.net/question/28_57183


Scheme语言--简介 
http://blog.csdn.net/zzulp/article/details/5547729


学习Scheme 
http://blog.csdn.net/ramacess/article/details/570769

 

Scheme 语言概要

http://www.ibm.com/developerworks/cn/linux/l-schm/index1.html

 

Read access to a .NET Assembly
https://ironscheme.codeplex.com/discussions/60977


Playing with IronScheme
http://www.codeproject.com/Articles/33050/Playing-with-IronScheme

尾递归

 http://blog.sina.com.cn/s/blog_3e5694650100tzqq.html

 

本文程序下载 “IronScheme中文版”

http://pwmis.codeplex.com/releases/view/117822

 

---------分界线 ----------------

 欢迎使用 PDF.NET SOD 开源框架,框架追求的目标是简单与效率的平衡,体现在:代码的精简,开发、维护的简单与追求极致的运行效率

作者简介:

本人现任职架构师,求伯乐,联系方式:http://www.pwmis.com/sqlmap

 

 

 

 

 

     本文转自深蓝医生博客园博客,原文链接:http://www.cnblogs.com/bluedoctor/p/3604293.html,如需转载请自行联系原作者


 

 

 

 

 

 




相关文章
|
编解码 JSON 网络协议
Golang 语言使用标准库 net/rpc/jsonrpc 包跨语言远程调用
Golang 语言使用标准库 net/rpc/jsonrpc 包跨语言远程调用
118 0
|
2月前
|
JSON C# 开发者
C#语言新特性深度剖析:提升你的.NET开发效率
【10月更文挑战第15天】C#语言凭借其强大的功能和易用性深受开发者喜爱。随着.NET平台的演进,C#不断引入新特性,如C# 7.0的模式匹配和C# 8.0的异步流,显著提升了开发效率和代码可维护性。本文将深入探讨这些新特性,助力开发者在.NET开发中更高效地利用它们。
37 1
|
5月前
|
人工智能 物联网 开发工具
.NET技术:多元语言、丰富库与跨平台能力引领软件开发新纪元。
`【7月更文挑战第4天】.NET技术:多元语言、丰富库与跨平台能力引领软件开发新纪元。从企业应用、云服务到游戏开发,其角色日益凸显。随着微软的持续创新与社区合作,未来.NET将在物联网、AI等领域拓宽应用,开发者应把握趋势,共创未来。`
44 0
|
7月前
|
C#
一个库帮你轻松的创建漂亮的.NET控制台应用程序
一个库帮你轻松的创建漂亮的.NET控制台应用程序
121 6
|
4月前
|
JSON C# 开发者
💡探索C#语言进化论:揭秘.NET开发效率飙升的秘密武器💼
【8月更文挑战第28天】C#语言凭借其强大的功能与易用性深受开发者喜爱。伴随.NET平台演进,C#持续引入新特性,如C# 7.0的模式匹配,让处理复杂数据结构更直观简洁;C# 8.0的异步流则使异步编程更灵活高效,无需一次性加载全部数据至内存。通过示例展示了模式匹配简化JSON解析及异步流实现文件逐行读取的应用。此外,C# 8.0还提供了默认接口成员和可空引用类型等特性,进一步提高.NET开发效率与代码可维护性。随着C#的发展,未来的.NET开发将更加高效便捷。
63 1
|
4月前
|
Unix Linux C#
增强用户体验:2个功能强大的.NET控制台应用帮助库
增强用户体验:2个功能强大的.NET控制台应用帮助库
|
6月前
|
存储 IDE C#
C#入门:在JetBrains Rider中创建.Net Framework控制台应用程序,输出“Hello, World!”
C#入门:在JetBrains Rider中创建.Net Framework控制台应用程序,输出“Hello, World!”
677 0
|
7月前
|
中间件 Go API
Golang深入浅出之-Go语言标准库net/http:构建Web服务器
【4月更文挑战第25天】Go语言的`net/http`包是构建高性能Web服务器的核心,提供创建服务器和发起请求的功能。本文讨论了使用中的常见问题和解决方案,包括:使用第三方路由库改进路由设计、引入中间件处理通用逻辑、设置合适的超时和连接管理以防止资源泄露。通过基础服务器和中间件的代码示例,展示了如何有效运用`net/http`包。掌握这些最佳实践,有助于开发出高效、易维护的Web服务。
88 1
|
Go
Golang 语言怎么使用 net/http 标准库开发 http 应用?
Golang 语言怎么使用 net/http 标准库开发 http 应用?
55 0
|
存储 SQL 监控
.NET开源简单易用、内置集成化的控制台、支持持久性存储的任务调度框架 - Hangfire
.NET开源简单易用、内置集成化的控制台、支持持久性存储的任务调度框架 - Hangfire