.NET Core跨平台执行命令、脚本的方法详细
一.前言
在工作中我们可能会遇到需要在程序中执行一些系统命令,来获取一些信息;或者调用shell脚本。.net core 目前已经可以跨平台执行,那么它如何跨平台执行命令呢,请看下面的讲解。话不多说了,来一起看看详细的介绍吧
二.processstartinfo、process 类介绍
我们主要用到的两个类就是 processstartinfo 和 process ,他们的用法和.net framework下是一样的。
1. processstartinfo 类
processstartinfo主要设置一些我们需要创建的进程的参数。比如需要启动的应用程序的文件名,参数等等。
(1)构造方法
它有三个构造方法:
public processstartinfo(); public processstartinfo(string filename); public processstartinfo(string filename, string arguments);
filename:用于启动进程的应用程序。
arguments:在进程启动时传递给应用程序的命令行参数。
(2)主要属性
createnowindow:指示是否在新窗口中启动进程。
redirectstandarderror:指示应用程序的错误输出是否写入到流中。
redirectstandardinput:指示是否从应用程序读取应用程序的输入流。
redirectstandardoutput:指示应用程序的文本输出是否写入流。
standarderrorencoding:错误输出内容编码。
standardoutputencoding:文本输出内容编码。
useshellexecute:指示是否使用操作系统shell启动进程。如果启动进程时使用shell,则为true; 如果应该直接从可执行文件创建进程,则为false。 默认值是true。
该类并没有定义自己的方法,因为它主要设置一些创建进程需要的参数信息。
2. process 类
该类的主要作用是提供对本地和远程进程的访问,并使你能够启动和停止本地系统进程。
(1).主要属性
exitcode:获取退出代码。0表示正常, 非0表示非正常退出。
exittime:获取关联进程退出的时间。
starttime:获取关联进程启动的时间。
hasexited:获取一个值,指示相关进程是否已终止。
machinename:获取运行关联进程的计算机的名称。
sessionid:获取关联进程的终端服务会话标识符。
standarderror:获取读取应用程序错误输出的流。
standardinput:获取应用程序输入内容的流。
standardoutput:获取用于读取应用程序文本输出的流。
threads:获取关联进程中正在运行的线程集合。
(2).主要方法
start :启动进程
beginerrorreadline:异步开始读取应用错误输出。
beginoutputreadline:异步开始读取应用标准输出。
cancelerrorread:取消读取错误输出。
canceloutputread:取消读取标准输出。
close:释放与此组件关联的所有资源。
closemainwindow:通过向其主窗口发送关闭消息来关闭具有用户界面的进程。
kill:立即停止关联的进程。
refresh:放弃已经在进程中缓存的关联进程的任何信息。
waitforexit:等待关联进程退出,可以设置超时时间,如不设置则一直等待。
(3)事件
一共有三个事件:
errordatareceived:接收到关联进程输出错误数据。
outputdatareceived:接收到关联进程输出标准数据。
exited:关联进程退出
三.在windows osx linux 下执行命令
这里我选择.net core带的 dotnet --info
输出.net core sdk&runtime相关的信息。
我们通过cmd执行会收到下面的信息:
1.编写代码执行命令
编写的代码如下:
static void main() { //创建一个processstartinfo对象 使用系统shell 指定命令和参数 设置标准输出 var psi = new processstartinfo("dotnet", "--info") {redirectstandardoutput = true}; //启动 var proc=process.start(psi); if (proc == null) { console.writeline("can not exec."); } else { console.writeline("-------------start read standard output--------------"); //开始读取 using (var sr = proc.standardoutput) { while (!sr.endofstream) { console.writeline(sr.readline()); } if (!proc.hasexited) { proc.kill(); } } console.writeline("---------------read end------------------"); console.writeline($"total execute time :{(proc.exittime-proc.starttime).totalmilliseconds} ms"); console.writeline($"exited code : {proc.exitcode}"); } }
执行结果如下:
从执行结果可以看出,我们通过编写的程序来执行dotnet --info
命令获取的结果几乎一样,只有第一行的提示,我们通过cmd执行命令输出的是中文,我们通过程序调用执行输出的是英文,这个问题,有兴趣的朋友可以研究一下。
2.在linux上执行
使用的系统环境为centos 7.2,.net core sdk版本为2.0.3。
直接执行命令结果如下:
我将代码上传到git server,然后在linux上clone然后执行结果如下:
可以看到我们获取执行输出是没有问题的,但是获取进程开始执行出错了,无法从进程检索该信息,现在我们移除统计执行时间的代码:
这下我们执行就没有问题了。从这里我们可以得出结论:由于平台的差异,获取一些信息可能会出现异常,所以我们实际一定要在多个平台上测试。
3.在osx上运行
我在osx上的.net core sdk版本为2.0.0 很久没更新了。
直接执行命令:
从git clone代码,执行结果如下:
可以看出我们在osx上执行是没有问题的。
四.在windows osx linux 下执行脚本1.编写测试脚本
编写脚本的主要逻辑为输出程序当前目录结构,然后输出一句话 “dotnet in 操作系统类型”
windows: win.bat
@echo off dir echo "dotnet in windows"
linux: linux.sh
#!/bin/bash ls echo "dotnet in linux"
osx: osx.sh
#!/bin/bash ls echo "dotnet in osx"
2.编写测试代码
我将所有的脚本都放在 项目根目录/shell 文件夹下。
因为我们需要根据不同的操作类型,选择不同的脚本来进行执行,所以我们需要在代码里面判断一下操作系统类型。我们可以通过 runtimeinformation.isosplatform
来判断。
static void main() { string filename="shell/"; //根据系统使用不同的shell文件 if (runtimeinformation.isosplatform(osplatform.windows)) { filename += "win.bat"; } else if (runtimeinformation.isosplatform(osplatform.linux)) { filename += "linux.sh"; } else { filename += "osx.sh"; } //创建一个processstartinfo对象 使用系统shell 指定命令和参数 设置标准输出 var psi = new processstartinfo(filename) { redirectstandardoutput = true }; //启动 var proc = process.start(psi); if (proc == null) { console.writeline("can not exec."); } else { console.writeline("-------------start read standard output--------------"); //开始读取 using (var sr = proc.standardoutput) { while (!sr.endofstream) { console.writeline(sr.readline()); } if (!proc.hasexited) { proc.kill(); } } console.writeline("---------------read end------------------"); console.writeline($"exited code : {proc.exitcode}"); } }
3.在windows下运行
在windows下运行是完全正常的。
4.在osx运行
直接运行会报一个权限异常,如下:
使用命令加入执行权限:
chmod +x osx.sh
然后再次执行:
可以看到成功执行了脚本。
5.在linux上运行
直接运行也是会有权限问题的:
同样使用命令加入执行权限:
chmod +x linux.sh
然后再次执行:
可以看到成功执行了我们的脚本。
4.容易犯的错误
看见上面的例子,我都成功执行了,其实我踩了几个坑,花了我不少时间来解决。
1.sh脚本一定要指定命令解析器
也就是这句话,放在sh脚本开头
#!/bin/bash
2.不管是windows linux osx 脚本编码必须为 ansi
不然程序执行的时候,读取字符会出错,造成执行异常。
五.写在最后
希望本文能给大家带来帮助,如有问题欢迎和我讨论。
本文所用代码地址:https://github.com/stulzq/blogdemos/tree/master/dotnetcmd
好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。
上一篇: 只要你有技术,程序员也能月薪过万
下一篇: 如何使用Swagger上传文件
推荐阅读