当前位置:   article > 正文

C#调用python函数

c#调用python函数

C#调用python函数的常用使用方法有利用Pythonnet、ironPython、打包exe、直接调用、打包成dll等方法


目录

1.IronPython:

安装

传参

第三方库

2.Pythonnet

将py文件编译成pyd文件

3.打包exe

3.1 生成exe

3.2 调用exe

3.3 传参

3.3.1 Json与C#之间的转换

3.3.2 Json与python之间的转换

3.4 传参过程中的问题

问题1:No module named 'XXX'

问题2:json串中含有引号

问题3:json串中含有空格

3.5 加快调用速度

4.直接调用

5.打包成DLL


1.IronPython:

ironPython3现在只在github上有源码,需要自己编译(我尝试失败),但是意外发现ironPython2也可以支持py3

安装

ironPython2的安装十分简单,安装和简单使用可以参考:.NET C# 调用python代码 简单实例_二等碗-CSDN博客_.net 调用python

也可以直接去官网下载,exe引导式安装,相当方便

传参

调用的代码参考:

C#:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.ComponentModel;
  4. using System.Data;
  5. using System.Drawing;
  6. using System.Linq;
  7. using System.Text;
  8. using System.Threading.Tasks;
  9. using System.Windows.Forms;
  10. using Microsoft.Scripting.Hosting;
  11. using IronPython.Hosting;
  12. namespace WindowsFormsApp8
  13. {
  14. public partial class Form1 : Form
  15. {
  16. public Form1() => InitializeComponent();
  17. private void button1_Click(object sender, EventArgs e)
  18. {
  19. if (textBox1.Text==string.Empty)
  20. {
  21. textBox1.Text = "0";
  22. }
  23. if (textBox2.Text == string.Empty)
  24. {
  25. textBox2.Text = "0";
  26. }
  27. int num1 = Convert.ToInt32(textBox1.Text);
  28. int num2 = Convert.ToInt32(textBox2.Text);
  29. //调用python
  30. ScriptRuntime pyRunTime = Python.CreateRuntime();
  31. dynamic obj = pyRunTime.UseFile("C:/Users/Lee/Desktop/States/test.py");
  32. label_result.Text = Convert.ToString(obj.add(num1,num2));
  33. }
  34. private void button2_Click(object sender, EventArgs e)
  35. {
  36. textBox1.Text = "0";
  37. textBox2.Text = "0";
  38. label_result.Text = "?";
  39. }
  40. }
  41. }

python:

  1. import sys
  2. def hello():
  3. return 'hello C#'
  4. def add(a, b):
  5. return (a + b)

但是ironPython配置简单,但是对py3支持的不是很好,对三方库的使用需要额外配置,并且不支持pyd文件

第三方库

具体如何配置第三方库可以参考:c#利用IronPython调用python的过程种种问题 - monkeyfx - 博客园

C#调用Python脚本并使用Python的第三方模块_dfskchfk72506的博客-CSDN博客

但是连接中使用的egg文件在相应位置是找不到的(如果你使用了pip或者conda安装的第三方库),因此你需要卸载该三方库并通过源码安装,

可以参考:python安装第三方库时生成.egg文件_曲終~的博客-CSDN博客


2.Pythonnet

参考:C#调用pyd - 灰信网(软件开发博客聚合)

要使用nuget模块,要求VS版本高一点,之前使用的VS2010不含此功能,2019可以。

将py文件编译成pyd文件

参考:python代码加密,打包pyd文件在C#中调用_星空的博客-CSDN博客_pyd文件

1.1安装Cython

pip install cython

1.2写好自己要编译的py函数文件test.py

  1. def LOVE(a,b):
  2. return a+b

1.3然后新建setup.py:

  1. from distutils.core import setup
  2. from Cython.Build import cythonize
  3. setup(ext_modules = cythonize("test.py"))#其中test.py是写好的待编译的文件

并在命令行里运行:

python setup.py build_ext --inplace

1.4 正常情况下会收到:

error: Unable to find vcvarsall.bat

如果没报错跳过这一步。

报错是因为VS缺少vcvarsall.bat文件,比较简单的方法是安装VS,但是VS版本和python有对应要求,我使用的是(VS2019,python3.9),还有一点要注意,安装VS时要把下面的选项勾选上:

1.5 重启电脑之后重新进行1.3步骤,会在py文件的同目录下的build\lib.win-amd64-3.9文件夹中得到对应的pyd文件test.cp39-win_amd64.pyd。

1.6 将test.cp39-win_amd64.pyd重命名与py文件相同test.pyd


3.打包exe

会将python的环境一起打包,调用时不需要安装python,但是运行速度会相对慢一点。

参考:https://blog.csdn.net/weixin_43944387/article/details/89945463?utm_medium=distribute.pc_relevant.none-task-blog-2%7Edefault%7EBlogCommendFromMachineLearnPai2%7Edefault-1.baidujs&dist_request_id=1328760.11392.16172458540053725&depth_1-utm_source=distribute.pc_relevant.none-task-blog-2%7Edefault%7EBlogCommendFromMachineLearnPai2%7Edefault-1.baidujs

c#调用python的方法总结_逸梵的博客-CSDN博客_c#调用python(这个链接实测可以跑通)

3.1 生成exe

参考:https://jingyan.baidu.com/article/ed2a5d1f03e60749f6be17d2.html

3.2 调用exe

这是我自己实现的代码

  1. using System;
  2. using System.Collections.Generic;
  3. using System.ComponentModel;
  4. using System.Data;
  5. using System.Drawing;
  6. using System.Linq;
  7. using System.Text;
  8. using System.Windows.Forms;
  9. using System.Diagnostics;
  10. namespace WindowsFormsApplication1
  11. {
  12. public partial class Form1 : Form
  13. {
  14. public Form1()
  15. {
  16. InitializeComponent();
  17. }
  18. private void button1_Click(object sender, EventArgs e)
  19. {
  20. label1.Text = "waiting . . .";
  21. string num1;
  22. if (textBox1.Text.Length != 0)
  23. num1 = textBox1.Text;
  24. else
  25. num1 ="0";
  26. string pyexePath = @".\temp.exe";
  27. Process p = new Process();
  28. p.StartInfo.FileName = pyexePath;//需要执行的文件路径
  29. p.StartInfo.UseShellExecute = false; //必需
  30. p.StartInfo.RedirectStandardOutput = true;//输出参数设定
  31. p.StartInfo.RedirectStandardInput = true;//传入参数设定
  32. p.StartInfo.CreateNoWindow = true;
  33. p.StartInfo.Arguments = num1;//参数以空格分隔,如果某个参数为空,可以传入””
  34. p.Start();
  35. string output = p.StandardOutput.ReadToEnd();
  36. p.WaitForExit();//关键,等待外部程序退出后才能往下执行}
  37. //Console.Write(output);//输出
  38. textBox1.Text = Convert.ToString(output);
  39. p.Close();
  40. label1.Text = "finish";
  41. }
  42. private void label1_Click(object sender, EventArgs e)
  43. {
  44. }
  45. private void textBox1_TextChanged(object sender, EventArgs e)
  46. {
  47. }
  48. }
  49. }

注意exe对应的路径,代码中是放在Debug目录下(debug模式运行)

3.3 传参

C#与python之间传参是通过String格式的参数传递,对于数组类型的参数特别不友好,为了解决这一问题,上面链接给出的一个解决方案是可以在C#中将数组写入文件,然后将文件地址传递给python,但是无端多了些许IO。传参只能以string的格式进行传递,因此可以借助Json进行参数传递。

但是,有一点要注意:

bash传参不支持引号

这就导致字典换后的Json在exe中转python会报错,而且报错的唯一反馈是:C#中会获取空值。

3.3.1 Json与C#之间的转换

这需要C#中的插件Newtonsoft.Json,可以在NuGet直接安装。

使用:

  1. List<double> list = new List<double>();
  2. list.Add(0.7);
  3. list.Add(0.6);
  4. list.Add(0.3);
  5. JsonSerializer serializer = new JsonSerializer();
  6. StringWriter sw = new StringWriter();
  7. serializer.Serialize(new JsonTextWriter(sw), Storage);
  8. string result = sw.GetStringBuilder().ToString();

就可以将list转化为result,通过调用exe这一小节中介绍的方法进行传参。

传参过程中可能会遇到多个参数类型不同的问题(字典也要求值类型相同)

为此,可以借助如何创建一个在C#中包含不同类型的字典? - 问答 - 云+社区 - 腾讯云介绍的HashTable

完整代码如下:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.ComponentModel;
  4. using System.Data;
  5. using System.Drawing;
  6. using System.Linq;
  7. using System.Text;
  8. using System.Windows.Forms;
  9. using System.Diagnostics;
  10. using System.Text;
  11. using System.IO;
  12. using System.Collections;
  13. using Newtonsoft.Json;
  14. namespace WindowsFormsApplication1
  15. {
  16. public static class StorageExtentions
  17. {
  18. public static T Get<T>(this Hashtable table, object key)
  19. {
  20. return (T)table[key];
  21. }
  22. }
  23. public partial class Form1 : Form
  24. {
  25. public Form1()
  26. {
  27. InitializeComponent();
  28. }
  29. private void button1_Click(object sender, EventArgs e)
  30. {
  31. int a1 = 12;
  32. string a2 = "test";
  33. double a3 = 24.1;
  34. double[] a4 = new double[]
  35. {
  36. 1,2.3,4.4,6.3,5.5,7.5
  37. };
  38. Hashtable Storage = new Hashtable();
  39. Storage.Add("age", a1);
  40. Storage.Add("name", a2);
  41. Storage.Add("bmi", a3);
  42. Storage.Add("grade", a4);
  43. JsonSerializer serializer = new JsonSerializer();
  44. StringWriter sw = new StringWriter();
  45. serializer.Serialize(new JsonTextWriter(sw), Storage);
  46. string result = sw.GetStringBuilder().ToString();
  47. MessageBox.Show(result);
  48. }
  49. }
  50. }

通过这种方式可以将所有参数转化为一个json数据。

3.3.2 Json与python之间的转换

这就比较简单,使用

json.loads和json.dumps函数即可,不再赘述。

但是还有一个eval函数,可以是适当使用。

3.4 传参过程中的问题

传参出问题很难调试,这是因为传参失败只会返回一个空值,不会有任何提示信息。这种C#调用exe实际上是通过命令行来调用,我们可以先通过在命令行中运行exe文件来判断错误出在exe中还是C#代码中,运行exe的命令为:

exe路径 参数1 参数2 参数3

路径可以是绝对路径或者是相对路径,参数就是exe中接收的参数。如,test.py代码如下,在其生成的exe中的参数有2个

  1. #test.py
  2. import sys
  3. def func(a, b):
  4. result = a + b
  5. return result
  6. if __name__ == '__main__':
  7. print(func(sys.argv[1], sys.argv[2]))

运行该exe的命令就是:test.exe 1 3 (当前处于test.exe所在目录)

问题1:No module named 'XXX'

  1. Traceback (most recent call last):
  2. File "Json_param.py", line 1, in <module>
  3. File "PyInstaller\loader\pyimod03_importers.py", line 531, in exec_module
  4. File "numpy\__init__.py", line 230, in <module>
  5. File "PyInstaller\loader\pyimod03_importers.py", line 531, in exec_module
  6. File "mkl\__init__.py", line 54, in <module>
  7. File "mkl\_mkl_service.pyx", line 27, in init mkl._py_mkl_service
  8. ModuleNotFoundError: No module named 'six'
  9. [8128] Failed to execute script Json_param

解决方案:pyinstaller打包python文件后,运行出现"Failed to execute script xxx.exe"错误的解决方式(No module named 'xxx')_武林大皮虾的博客-CSDN博客

six模块虽然没在代码中使用,但是间接使用,但是pyinstaller打包时并未指定,所以执行时找不到此模块

解决方法:生成exe时生成了一个同名spec文件

在文件的hiddenimports中写入‘six’,然后用该文件重新生成一次exe文件,pyinstaller test.spec。

还有另一种解决方案,pyinstaller打包遇到的一些坑_HQ1356466973的博客-CSDN博客

如果这个东西不是程序必须得,那么打包的时候加入--hidden-import  库名.xxx.xxx,不导入它即可解决:pyinstaller -Fw yourfile.py --hidden-import 库名.xxx.xxx

但是我没有验证过。

另外一种简单的方法是直接在要运行的函数中import缺失的模块(亲测好用)

问题2:json串中含有引号

引号在json中不能传递,转义也不行,现在的解决方法是在传递之前replace替换成指定字符串(我用的是 shuangyinhao_Archer),传递完成后在替换回来,有更好的方法希望不吝赐教。

传参可以同时传多个参数,因此,更好的办法是将每个参数分别Json编码,这样基本不会有引号的问题,但是空格问题还是要单独处理。这里Json编码的作用是保持数组等数据格式,在Python中直接使用eval函数就可以得到原来的形式。

问题3:json串中含有空格

命令行相当不智能,区分不同参数的唯一方法是用空格,因此,在生成json串之前替换成指定字符串或者直接去掉(字典和类中的空格去掉影响不大)

3.5 加快调用速度

https://blog.csdn.net/u014421797/article/details/103302925?utm_medium=distribute.pc_relevant_t0.none-task-blog-2%7Edefault%7EBlogCommendFromMachineLearnPai2%7Edefault-1.baidujs&dist_request_id=1328769.14893.16173739685530525&depth_1-utm_source=distribute.pc_relevant_t0.none-task-blog-2%7Edefault%7EBlogCommendFromMachineLearnPai2%7Edefault-1.baidujs

4.直接调用

参考:C# 调用 Python(直接调用py代码方式)_coco的专栏-CSDN博客

5.打包成DLL

参考:手把手教你将Python程序打包为DLL_zmr1994的博客-CSDN博客_python生成dll

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/从前慢现在也慢/article/detail/478497
推荐阅读
相关标签
  

闽ICP备14008679号