使用IronPython把Python脚本集成到.NET程序中的教程


Posted in Python onMarch 31, 2015

从两个优秀的世界各取所需,更高效的复用代码。想想就醉了,.NET和python融合了。“懒惰”的程序员们,还等什么?

Jesse Smith为您展示如何两个语言来服务同一个.NET程序。你能集两家所长:Python和.NET一起工作,提供可重用的代码功能而不需要你为了一个环境重写代码库。

通过使用IronPython 运行时库,你可以让Python脚本运行在你的.NET程序中。本文向你展示如何使用一个.NET程序中的python脚本获取并展示用户反馈。

如果你曾经有在一个.NET程序中运行Python脚本的需求,整合两者最好的办法是使用IronPython。我有过这样的需求。我曾经效力的一个组织需要扩展一个已经存在于一个.NET地图程序中的工具集。我们的目的是利用另一个工具集中现有的Python代码来扩展这个已有的工具集,前者还不是后者的一部分。在做了一番研究之后,我决定使用IronPython,事情进展很顺利。

本文基于上述项目的经验,我会还原当时的场景,对你所处的情境也同样适用。

对于这个我们所期望的解决方案,首先要面对的挑战是从将要运行在.NET程序中的那个Python脚本获取输出,然后使用一个.NET程序的对话框把这个输出展示给用户。让我感到吃惊的是,这么做比预想的还要简单,你同样也能学到如何去做同样的事情。整个过程相对比较简单,我们将在下面的章节中进行概述。
在.NET程序中嵌入Python脚本

在这个例子中,我向你展示如何在一个Windows桌面程序中嵌入Python脚本。目的是在一个已有的程序中添加一个新的工具按钮。点击按钮后,程序会弹出一个对话框,显示一个嵌入在对话框代码中的Python脚本的运行结果。

为了给这个例子铺垫更多的前因后果,我们这个应用程序是一个桌面地图软件,可以让用户创建他们自己的地图。这个新的工具允许用户解析和标准化他们地图上的地址点。

标准化一个地址,就意味着要确保这个地址有一个街道名,前缀或后缀,房屋或建筑编号,以及连接该街道的上一条街道和下一条街道。每一个地址段必须遵循US邮局邮编地址标准指导。

假设已有一个Python脚本已经可以做到这些;它就是例子中执行解析的脚本。Python脚本将会通过当前对话框在屏幕上输出或闪现这个正在解析的地址,我们可以通过选择地址解析工具来触发对话框(我不想说的太细,仅仅解释到这里。)

代码并不重要,重要的是知道如何嵌入脚本和如何定向脚本的输出到对话框,作为程序的一部分显示在屏幕上。这一过程像下面这样进行:

  •     用户启动地图程序并打开一个自定义地图。
  •     用户从屏幕上方的已有的工具集中选择一个新的地址标准化工具。
  •     弹出一个带有可以启动地址标准化进程的启动按钮的地址标准化对话框
  •     一个文本框显示当前正在被解析的地址。这些文本框很快消失,地址在用户眼前一闪而过,表示过程启动并且正在解析。
  •     出现一个表示整个过程结束的信息,使用了第4步中同样的文本框。

确定源代码和项目

首先要做的事情是确定脚本应该嵌入到应用程序源代码的什么位置。在我们的例子/情境下,这个位置应该是添加到包含工具集项目中的新的对话框。这个对话框会被一个已经存在的用于处理工具栏点击事件的方法所触发

当我们确定了需要包含源代码的项目后,就需要引用IronPython 库。

如果你使用Visual Studio作为编辑器的话,最简单的方法是使用Nuget Package Manager来添加IronPython库到项目中。你可以搜索”IronPython”,然后运行时库就可以在包管理工具中选择了。
嵌入脚本

接下来做的才是真正嵌入脚本。你首先要用脚本引擎(scripting engine)来设置一个Python脚本引擎实例。在添加脚本前,你同样可以设置任何你的脚本所需的特殊的路径。

ScriptEngine pyEngine = Python.CreateEngine();
pyEngine.Runtime.IO.RedirectToConsole();
var paths = pyEngine.GetSearchPaths();
paths.Add(@"C:Python27Lib");
paths.Add(@"C:Python27Libsite-packages");
pyEngine.SetSearchPaths(paths);

第二行告诉.NET框架Python引擎的运行时库会将输出重定向到控制台。然而,这并不是重定向到我们为这个应用添加的新的工具所需的对话框中。(下面的代码会做这个工作)

但是,首先我们需要通过一个简单的字符串变量来添加脚本。你需要改变你的脚本中的引号来配合字符串的引号工作。

一个简单的方法是把你脚本中的所有双引号变成单引号。嵌入脚本的语法如下:

string thescript =
  @"
  (此处为实际脚本内容)
  ";

你可能需要处理一些格式问题,但是缩进必须一致。在脚本字符串解析一个有效的字符串后,是时候添加输出重定向代码让脚本的输出显示在工具的对话框窗口中了:

Console.SetOut(TextWriter.Synchronized(new TextBoxWriter(statusText)));
    pyEngine.Execute(thescript);
    this.AllDone(FINISHED);
   }
   catch (Exception ex)
   {
    this.AllDone(ex.InnerException.StackTrace);
   }
  }
  public void AllDone(string message)
  {
   buttonStart.Enabled = true;
   this.statusLabel.Text = message;
  }

在上面这段代码中,我们设置了一个新的TextWriter,它接受一个TextBoxWriter类型的参数,这使得我们可以把脚本的输出重新写到一个文本框中。TextBoxWriter类型的代码如下:
 

public class TextBoxWriter : TextWriter
  {
   private TextBox _textBox;
   public TextBoxWriter(TextBox textbox)
   {
    _textBox = textbox;
   }
   public override void Write(char value)
   {
    base.Write(value);
    // When character data is written, append it to the text box.
    _textBox.AppendText(value.ToString());
   }
   public override System.Text.Encoding Encoding
   {
    get { return System.Text.Encoding.UTF8; }
   }
  }
 }

传入TextBoxWriter类型的statusText属性是我们的文本框,它会出现在对话框中,显示脚本的输出。我们的Python脚本中的每一个输出语句都会被重定向到这个文本框。
结论

在本文中,你学会了如何把Python脚本集成到一个.NET程序中,并且把Python的脚本文件输出到一个.NET对话框。这种无缝衔接,用户是不会感觉到的,他们并不知道实际上是Python在处理后台的一些工作。

在很多场合下,集成两种语言是很有用的。我分享了的这一情景,为我的处境提供了很好的解决方案。你可以按照类似的步骤,用同样的方法把它应用在很多场合。

我建议你去建立一个自己的简单范例,甚至使用Python脚本文件把Python代码直接加入到.NET应用中,你确实可以这样做。当然你并不需要直接把脚本嵌入在.NET源代码中,但对我来说这样做最方便。

Python 相关文章推荐
利用Django-environ如何区分不同环境
Aug 26 Python
Python初学者需要注意的事项小结(python2与python3)
Sep 26 Python
python通过ffmgep从视频中抽帧的方法
Dec 05 Python
Python面向对象总结及类与正则表达式详解
Apr 18 Python
Python3+OpenCV2实现图像的几何变换(平移、镜像、缩放、旋转、仿射)
May 13 Python
Django后端接收嵌套Json数据及解析详解
Jul 17 Python
Python定时任务工具之APScheduler使用方式
Jul 24 Python
通过实例解析python描述符原理作用
Jan 22 Python
Django模板标签中url使用详解(url跳转到指定页面)
Mar 19 Python
python logging.info在终端没输出的解决
May 12 Python
python使用matplotlib绘制折线图的示例代码
Sep 22 Python
Python使用cn2an实现中文数字与阿拉伯数字的相互转换
Mar 02 Python
提升Python程序运行效率的6个方法
Mar 31 #Python
用Python从零实现贝叶斯分类器的机器学习的教程
Mar 31 #Python
利用Python的Flask框架来构建一个简单的数字商品支付解决方案
Mar 31 #Python
用Python进行基础的函数式编程的教程
Mar 31 #Python
python使用多线程不断刷新网页的方法
Mar 31 #Python
Python新手实现2048小游戏
Mar 31 #Python
举例介绍Python中的25个隐藏特性
Mar 30 #Python
You might like
饭制《星际争霸》Mod:优化游戏机制 增加新单位
2017/07/02 星际争霸
轻松修复Discuz!数据库
2008/05/03 PHP
解析php mysql 事务处理回滚操作(附实例)
2013/08/05 PHP
深入分析PHP引用(&)
2014/09/04 PHP
初窥JQuery-Jquery简介 入门了解篇
2010/11/25 Javascript
JavaScript加入收藏夹功能(兼容IE、firefox、chrome)
2014/05/05 Javascript
Javascript验证用户输入URL地址是否为空及格式是否正确
2014/10/09 Javascript
纯javascript实现分页(两种方法)
2015/08/26 Javascript
JS实现网页标题栏显示当前时间和日期的完整代码
2015/11/02 Javascript
jquery实现的点击翻书效果代码
2015/11/04 Javascript
详解JavaScript基于面向对象之创建对象(1)
2015/12/10 Javascript
Three.js学习之文字形状及自定义形状
2016/08/01 Javascript
简单实现js浮动框
2016/12/13 Javascript
JavaScript实现前端实时搜索功能
2020/03/26 Javascript
详解Vue.js 2.0 如何使用axios
2017/04/21 Javascript
node.js基于socket.io快速实现一个实时通讯应用
2019/04/23 Javascript
Vue-Cli 3.0 中配置高德地图的两种方式
2019/06/19 Javascript
JavaScript Event Loop相关原理解析
2020/06/10 Javascript
Vue封装Axios请求和拦截器的步骤
2020/09/16 Javascript
JS实现选项卡插件的两种写法(jQuery和class)
2020/12/30 jQuery
Python中random模块用法实例分析
2015/05/19 Python
Python 通过pip安装Django详细介绍
2017/04/28 Python
python简单商城购物车实例代码
2018/03/15 Python
python 函数内部修改外部变量的方法
2018/12/18 Python
python画图常规设置方式
2020/03/05 Python
如何使用PyCharm将代码上传到GitHub上(图文详解)
2020/04/27 Python
丝芙兰巴西官方商城:SEPHORA巴西
2016/10/31 全球购物
名词解释WEB SERVICE,SOAP,UDDI,WSDL,JAXP,JAXM;JSWDL开发包的介绍。
2012/10/27 面试题
怎样声明接口
2014/09/19 面试题
银行实习人员自我鉴定
2013/09/22 职场文书
医科大学生的自我评价
2013/12/04 职场文书
班组拓展活动方案
2014/08/14 职场文书
法定代表人证明书
2014/11/28 职场文书
追悼会家属答谢词
2015/09/29 职场文书
深入理解redis中multi与pipeline
2021/06/02 Redis
vue ref如何获取子组件属性值
2022/03/31 Vue.js