IronPython和C#交互

IronPython是一个.NET平台上的Python实现,包括了完整的编译器、执行引擎与运行时支持,能够与.NET已有的库无缝整合到一起。

IronPython已经很好的集成到了.NET framework中,所以Ironpython和C#的交互也就变得很简单了。下面就通过一些简单的例子来看看IronPython和C#之间的交互。

环境设置

工欲善其事,必先利其器,所以在开始IronPython的开发之前,我们先找到一个方便的开发环境。

PTVS(Python tools for Visual Studio)是一个免费开源的VisualStudio的插件,支持 VisualStudio 2010/2012/2013,安装好这个插件之后,我们就可以直接通过VS进行IronPython的开发了。

下面一个截图显示了我们可以新建的项目模板:

IronPython调用C#

首先我们看下如何在IronPython中使用C#的简单例子。

使用标准.NET库

在.NET中,有很多标准库,在IronPython中,就可以使用import来引入这些标准库来直接使用。看一个简单的例子,我们使用.NET中的String和DateTime

from System import DateTime, String
formatStr = String.Format("{0} {1}", "Hello World! The current date and time is ", DateTime.Now)
print formatStr
print dir(String)
raw_input("press Enter to exit!")

代码输出如下,可以看到在IronPython代码中,可以通过String的Format方法进行字符串格式化的输出。

引入.NET库

在.NET开发中,会经常通过References来引用一些.NET库,当然在IronPython项目中,也可以引用并使用.NET库。

例如,现在我们有一个Calc的计算类型,里面有一个Add和Sub方法。通过这个类型,生成了一个CalcLib.dll。

namespace CalcLib
{
public class Calc
{
public int Add(int a, int b)
{
return a + b;
} public int Sub(int a, int b)
{
return a - b;
}
}
}

下面看看如何在IronPython项目中使用这个dll,在IronPython中,可以使用"clr"模块来添加.NET引用:

import clr
clr.AddReference('CalcLib')
#clr.AddReferenceToFile('CalcLib.dll')
from CalcLib import Calc
print dir(Calc)
calcObj = Calc()
print "result of 3+4 is:", calcObj.Add(3,4)
print "result of 10+2 is:", calcObj.Sub(10,2) raw_input("press Enter to exit!")

代码输出如下,当引用了CalcLib.dll之后,我们就可以使用Calc类型创建实例,并且使用实例的C#方法。

IronPython创建WPF应用

在IronPython项目中,也可以引入WPF相关的.NET库,这样就可以方便的创建图形界面应用。

安装过PTVS之后,里面有个"IronPython WPF Application"的模板,通过这个模板,可以直接创建WPF应用。

在新建的项目中,VS会帮我们自动引用WPF相关的库,同时会有一个.py和.xaml文件。

下面看一个简单的例子,通过IrpnPython实现的一个简单计算器,界面的代码如下:

<Grid>
<Grid.RowDefinitions>
<RowDefinition></RowDefinition>
<RowDefinition></RowDefinition>
<RowDefinition></RowDefinition>
<RowDefinition></RowDefinition>
<RowDefinition></RowDefinition>
<RowDefinition></RowDefinition>
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition></ColumnDefinition>
<ColumnDefinition></ColumnDefinition>
<ColumnDefinition></ColumnDefinition>
<ColumnDefinition></ColumnDefinition>
</Grid.ColumnDefinitions> <TextBlock Name="InputTb" Grid.Row="0" Grid.ColumnSpan="4"/>
<TextBlock Name="ResultTb" Grid.Row="1" Grid.ColumnSpan="3"/> <Button Content="1" Grid.Row="2" Grid.Column="0" Click="Input_Button_Click"/>
<Button Content="2" Grid.Row="2" Grid.Column="1" Click="Input_Button_Click"/>
<Button Content="3" Grid.Row="2" Grid.Column="2" Click="Input_Button_Click"/>
<Button Content="4" Grid.Row="3" Grid.Column="0" Click="Input_Button_Click"/>
<Button Content="5" Grid.Row="3" Grid.Column="1" Click="Input_Button_Click"/>
<Button Content="6" Grid.Row="3" Grid.Column="2" Click="Input_Button_Click"/>
<Button Content="7" Grid.Row="4" Grid.Column="0" Click="Input_Button_Click"/>
<Button Content="8" Grid.Row="4" Grid.Column="1" Click="Input_Button_Click"/>
<Button Content="9" Grid.Row="4" Grid.Column="2" Click="Input_Button_Click"/>
<Button Content="0" Grid.Row="5" Grid.Column="0" Click="Input_Button_Click"/>
<Button Content="+" Grid.Row="2" Grid.Column="3" Click="Input_Button_Click"/>
<Button Content="-" Grid.Row="3" Grid.Column="3" Click="Input_Button_Click"/>
<Button Content="*" Grid.Row="4" Grid.Column="3" Click="Input_Button_Click"/>
<Button Content="/" Grid.Row="5" Grid.Column="3" Click="Input_Button_Click"/>
<Button Content="." Grid.Row="5" Grid.Column="1" Click="Input_Button_Click"/> <Button Content="C" Grid.Row="5" Grid.Column="2" Click="Clear_Button_Click"/> <Button Content="=" Grid.Row="1" Grid.Column="3" Click="Calc_Button_Click"/>
</Grid>

对应的IronPython代码如下:

from __future__ import division
import traceback
import wpf from System.Windows import Application, Window, MessageBox class MyWindow(Window):
def __init__(self):
wpf.LoadComponent(self, 'IronPythonWPF.xaml') def Calc_Button_Click(self, sender, e):
try:
result = eval(self.InputTb.Text)
self.ResultTb.Text = str(result)
except Exception, e:
tracelog = traceback.format_exc()
MessageBox.Show(str(e)) pass def Clear_Button_Click(self, sender, e):
self.InputTb.Text = ""
self.ResultTb.Text = ""
pass def Input_Button_Click(self, sender, e):
self.InputTb.Text += sender.Content
pass if __name__ == '__main__':
Application().Run(MyWindow())

代码运行效果如下:

C#调用IronPython

前面介绍了在IronPython中如何使用.NET库,下面看看通过C#代码执行IronPython脚本。在.NET framework中,包含了IronPython的编译器和执行引擎,所以我们可以通过C#代码创建一个引擎实例,然后执行脚本。

先看看我们需要使用的类型:

  • ScriptEngine: 动态语言(IronPython)执行类,可于解析和执行动态语言代码。
  • ScriptScope:构建一个执行上下文,其中保存了环境及全局变量;宿主(Host)可以通过创建不同的 ScriptScope 来提供多个数据隔离的执行上下文。
  • ScriptSource:操控动态语言代码的类型,可以编译(Compile)、运行(Execute)代码。

现在我们有一个简单的打印当前时间的IronPython脚本:

import datetime
print "current datetiem is:", datetime.datetime.now()

然后就可以使用下面的方式执行脚本:

static void Main(string[] args)
{
try
{
ScriptEngine engine = Python.CreateEngine();
ScriptScope scope = engine.CreateScope(); ScriptSource script = engine.CreateScriptSourceFromFile(@"Script.py"); var result = script.Execute(scope);
}
catch (Exception e)
{
Console.WriteLine(e.Message);
} Console.Read();
}

给IronPython传递参数

在ScriptScope类型中,有一个SetVariable方法,我们可以通过这个方法给脚本传递参数。

public void SetVariable(string name, object value)

这样,我们就可以把一个C#实例传递给IronPython,然后脚本就可以使用C#实例的成员。看一个例子:

public class Student
{
public int Age { get; set; }
public string Name { get; set; }
public override string ToString()
{
return string.Format("{0} is {1} years old", this.Name, this.Age);
}
} class Program
{
static void Main(string[] args)
{
try
{
ScriptEngine engine = Python.CreateEngine();
ScriptScope scope = engine.CreateScope();
Student stu = new Student { Name = "Wilber", Age = 28 };
scope.SetVariable("stuObj", stu);
ScriptSource script = engine.CreateScriptSourceFromFile(@"PrintStuInfo.py"); var result = script.Execute(scope); }
catch (Exception e)
{
Console.WriteLine(e.Message);
} Console.Read();
}
}

在这个例子中,C#代码中创建了一个Student类型的实例,并把这个实例传递给了PrintStuInfo.py脚本。

print "Student name:", stuObj.Name
print "Student age:", stuObj.Age
print stuObj.ToString()

通过输出可以看到,IronPython脚本可以方便的访问C#实例的成员。

总结

本篇文章通过一些例子演示了IronPython与C#的交互,感觉有几个例子还是很有意思的。

有时候使用C#调用IronPython可以使程序变得更加灵活,通过一个C#类型提供一组封装好的操作,每次构建类型实例然后传递给脚本;这样,用户就可以编写IronPython脚本,然后使用C#类型中提供的操作方法,从而实现不同的自定义操作。

 
分类: IronPython

IronPython和C#交互的更多相关文章

  1. IronPython 与C#交互

    http://www.cnblogs.com/nuaalfm/archive/2010/02/11/1667448.html 一.介绍 Python是一种面向对象.直译式计算机程序设计语言,也是一种功 ...

  2. 转Python 和C#的交互

    http://www.cnblogs.com/wilber2013/category/708919.html IronPython和C#交互   IronPython是一个.NET平台上的Python ...

  3. C#Note13:如何在C#中调用python

    前言 IronPython 是一种在 .NET 及 Mono上的 Python 实现,由微软的 Jim Hugunin(同时也是 Jython 创造者) 所发起,是一个开源的项目,基于微软的 DLR ...

  4. C#与Python交互方式

    前言: 在平时工作中,需求有多种实现方式:根据不同的需求可以采用不同的编程语言来实现.发挥各种语言的强项 如:Python的强项是:数据分析.人工智能等 .NET 开发桌面程序界面比Python更简单 ...

  5. 工大助手(C#与python交互)

    工大助手(爬虫--C#与python交互) 基本内容 工大助手(桌面版) 实现登陆.查成绩.计算加权平均分等功能 团队人员 13070046 孙宇辰 13070003 张帆 13070004 崔巍 1 ...

  6. 使用IronPython给.Net程序加点料

    开发的时候,经常被策划频繁变动的方案而苦恼.这时候就想要加入点动态语言来辅助一下. 在考虑用动态语言之前也曾想过使用动态加载dll的方式,实现基础接口来调用.在卸载的时候遇到了问题,虽可以通过应用程序 ...

  7. IronPython脚本调用C#dll示例

    上篇Python脚本调用C#代码数据交互示例(hello world)介绍了与C#紧密结合的示例,这里还将提供一个与C#结合更紧密的示例,直接调用C#编写的DLL.      我们还是沿用了上篇文章的 ...

  8. Python脚本调用C#代码数据交互示例(hello world)

    原地址: http://www.djangochina.cn/forum.php?mod=viewthread&tid=247 随着项目的逐渐收尾, 对IronPython脚本也越来越熟悉,这 ...

  9. IronPython 源码剖析系列(2):IronPython 引擎的运作流程

    http://blog.csdn.net/inelm/article/details/4612987 一.入口点 Python 程序的执行是从 hosting 程序 ipy.exe 开始的,而他的入口 ...

随机推荐

  1. Cocos2d-x场景变化相关功能介绍

    现场由导演级交换机Director实现.之间的相关的功能,如下面: runWithScene(Scene* scene).该函数能够执行场景.仅仅能在启动第一个场景时候调用该函数.假设已经有一个场景执 ...

  2. HDU 4686 Arc of Dream(递归矩阵加速)

    标题效果:你就是给你一程了两个递推公式公式,第一个让你找到n结果项目. 注意需要占用该公式的复发和再构造矩阵. Arc of Dream Time Limit: 2000/2000 MS (Java/ ...

  3. OAuth在WebApi

    OAuth在WebApi中的使用,前后台分离的调用方式 前段时间由于公司架构服务层向WebApi转换,就研究了OAuth在WebApi中的使用,这中间遇到了很多坑,在此记录一下OAuth的正确使用方式 ...

  4. Paypal-Express Checkout快捷支付方式的android端开发心得(二)

    一.前导 上一篇讲的不是非常好,这里再又一次讲一下. Paypal手机支付有2种形式: 1.Mobile Express Checkout,MEC,快捷支付 2.MPL 假设採用MEC支付方式,这样的 ...

  5. 使用SQLServer 2008的CDC功能实现数据变更捕获

    原文:使用SQLServer 2008的CDC功能实现数据变更捕获 最近由于工作需要,研究了一下2008 CDC功能,觉得还不错,下面整理了一下研究过程,虽然比较粗略,但是基本上能用了,如果有补充请大 ...

  6. SGU 200. Cracking RSA(高斯消元+高精度)

    标题效果:鉴于m整数,之前存在的所有因素t素数.问:有多少子集.他们的产品是数量的平方. 解题思路: 全然平方数就是要求每一个质因子的指数是偶数次. 对每一个质因子建立一个方程. 变成模2的线性方程组 ...

  7. 理解git经常使用命令原理

    git不同于类似SVN这样的版本号管理系统,尽管熟悉经常使用的操作就能够满足大部分需求,但为了在遇到麻烦时不至于靠蛮力去尝试,了解git的原理还是非常有必要. 文件 通过git管理的文件版本号信息所有 ...

  8. 十依据一个有用的算法来找到最小(最大)的k的数量-线性搜索算法

    例如:进入1.2.3,4,5,6.7.8此8数字,最小的4图的1,2,3,4. 思路1:最easy想到的方法:先对这个序列从小到大排序.然后输出前面的最小的k个数就可以.假设选择高速排序法来进行排序, ...

  9. Linux概念架构的理解(转)

    英文原文:Conceptual Architecture of the Linux Kernel 摘要 Linux kernel成功的两个原因:(1)架构设计支持大量的志愿开发者加入到开发过程中:(2 ...

  10. ubuntu经常使用的命令摘要

    1.df命令 # df -ha 显示所有文件和分区的使用 # df -h /dev/sda1 显示sda1磁盘使用率 # df -T 显示文件系统名称属于每个分区.区的格式类型(比方ext3) 注:h ...