一个C#写的调用外部进程类

2008-05-21 07:00 作者: 肖波 出处: 天极网

 C# 调用外部进程的类,网上可以搜出很多来,为什么要再写一遍,实在是因为最近从网上拷贝了一个简单的例程用到项目中,运行有问题,后来研究了半天,才解决了这些问题。于是打算重写,一来说说调用一个外部进程这么简单的一件事究竟会有哪些问题,二来也希望我写的这个相对比较完整的类可以为软件开发的同道们节约一些脑细胞,以便集中优势兵力解决那些真正高深复杂的软件问题。

  在开始正题之前,我们先来看一看网上比较常见的执行外部进程的函数

  


private string RunCmd(string command)
   {
   //例Process
   Process p = new Process();
  
   p.StartInfo.FileName = "cmd.exe"; //确定程序名
   p.StartInfo.Arguments = "/c " + command; //确定程式命令行
   p.StartInfo.UseShellExecute = false; //Shell的使用
   p.StartInfo.RedirectStandardInput = true; //重定向输入
   p.StartInfo.RedirectStandardOutput = true; //重定向输出
   p.StartInfo.RedirectStandardError = true; //重定向输出错误
   p.StartInfo.CreateNoWindow = true; //设置置不显示示窗口
  
   p.Start(); //00
  
   //p.StandardInput.WriteLine(command); //也可以用这种方式输入入要行的命令
   //p.StandardInput.WriteLine("exit"); //要得加上Exit要不然下一行程式
  
   return p.StandardOutput.ReadToEnd(); //输出出流取得命令行结果
   }
  

  这个方法应该是比较常见的调用外部进程的方法,我以前也一直是这样调用外部进程的,也没有碰到过什么问题。但这次调用的外部进程比较特殊,用这种方法调用就出现了两个问题。

  第一个问题是这个被调用的外部进程有时候会出现异常,出现异常后Windows会弹出错误报告框,程序于是吊死在那里,必须手工干预。这个问题比较好解决,程序中设置一下注册表搞定。

  第二个问题是调用这个外部进程(是一个控制台进程)后,程序会阻塞在p.StandardOutput.ReadToEnd();这一句,永远无法出来,被调用的那个控制台程序也被吊死。但该控制台进程在CMD 中是可以正常执行的。后来看来一些资料才发现原来原因是出在该控制台程序控制台输出大量字符串,管道重定向后,调用程序没有及时将管道中的输出数据取出,结果导致管道被阻塞,程序吊死。在这里还有另外一个问题,虽然这次没有遇到,但网上有其他人遇到,就是错误信息管道不及时取出数据,也会被阻塞,而且如果要同时取出两个管道的数据,必须要利用一个辅助线程才能实现。

  问题讲完了,下面给出这个类的完整代码

  


using System;
  using System.Collections.Generic;
  using System.Text;
  using System.Runtime.InteropServices;
  using System.Threading;
  
  namespace Laboratory.Process
  {
   class ReadErrorThread
   {
   System.Threading.Thread m_Thread;
   System.Diagnostics.Process m_Process;
   String m_Error;
   bool m_HasExisted;
   object m_LockObj = new object();
  
   public String Error
   {
   get
   {
   return m_Error;
   }
   }
  
   public bool HasExisted
   {
   get
   {
   lock (m_LockObj)
   {
   return m_HasExisted;
   }
   }
  
   set
   {
   lock (m_LockObj)
   {
   m_HasExisted = value;
   }
   }
   }
  
   private void ReadError()
   {
   StringBuilder strError = new StringBuilder();
   while (!m_Process.HasExited)
   {
   strError.Append(m_Process.StandardError.ReadLine());
   }
  
   strError.Append(m_Process.StandardError.ReadToEnd());
  
   m_Error = strError.ToString();
   HasExisted = true;
   }
  
   public ReadErrorThread(System.Diagnostics.Process p)
   {
   HasExisted = false;
   m_Error = "";
   m_Process = p;
   m_Thread = new Thread(new ThreadStart(ReadError));
   m_Thread.Start();
   }
  
   }
  
   class RunProcess
   {
   private String m_Error;
   private String m_Output;
  
   public String Error
   {
   get
   {
   return m_Error;
   }
   }
  
   public String Output
   {
   get
   {
   return m_Output;
   }
   }
  
   public bool HasError
   {
   get
   {
   return m_Error != "" && m_Error != null;
   }
   }
  
   public void Run(String fileName, String para)
   {
   StringBuilder outputStr = new StringBuilder();
  
   try
   {
   //disable the error report dialog.
   //reference: http://www.devcow.com/blogs/adnrg/archive/2006/07/14/Disable-Error-Reporting-Dialog-for-your-application-with-the-registry.aspx
   Microsoft.Win32.RegistryKey key;
   key = Microsoft.Win32.Registry.LocalMachine.OpenSubKey(@"software\microsoft\PCHealth\ErrorReporting\", true);
   int doReport = (int)key.GetValue("DoReport");
  
   if (doReport != 0)
   {
   key.SetValue("DoReport", 0);
   }
  
   int showUI = (int)key.GetValue("ShowUI");
   if (showUI != 0)
   {
   key.SetValue("ShowUI", 0);
   }
   }
   catch
   {
   }
  
  
   m_Error = "";
   m_Output = "";
   try
   {
   System.Diagnostics.Process p = new System.Diagnostics.Process();
  
   p.StartInfo.FileName = fileName;
   p.StartInfo.Arguments = para;
   p.StartInfo.UseShellExecute = false;
   p.StartInfo.RedirectStandardInput = true;
   p.StartInfo.RedirectStandardOutput = true;
   p.StartInfo.RedirectStandardError = true;
   p.StartInfo.CreateNoWindow = true;
  
   p.Start();
  
   ReadErrorThread readErrorThread = new ReadErrorThread(p);
  
   while (!p.HasExited)
   {
   outputStr.Append(p.StandardOutput.ReadLine()+"\r\n");
   }
  
   outputStr.Append(p.StandardOutput.ReadToEnd());
  
   while (!readErrorThread.HasExisted)
   {
   Thread.Sleep(1);
   }
  
   m_Error = readErrorThread.Error;
   m_Output = outputStr.ToString();
   }
   catch (Exception e)
   {
   m_Error = e.Message;
   }
   }
  
   }
  }
  

转载于:https://www.cnblogs.com/gofficer/archive/2008/10/30/1322825.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/264487.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【编程题目】复杂链表的复制☆

76.复杂链表的复制(链表、算法)题目:有一个复杂链表,其结点除了有一个 m_pNext 指针指向下一个结点外,还有一个 m_pSibling 指向链表中的任一结点或者 NULL。其结点的 C定义如下:struct ComplexNode{ int m…

mssql sqlserver 不固定行转列数据(动态列)

mssql sqlserver 不固定行转列数据(动态列) 原文:mssql sqlserver 不固定行转列数据(动态列)转自:http://www.maomao365.com/?p5471摘要: 下文主要讲述动态行列转换语句,列名会根据行数据的不同, 动态的发生变化 -------------------------------------…

在php里让字体划过变色,鼠标划过字体时如何用css来实现字体变色?(代码实测)...

当我们在浏览网页时,鼠标划过某段文字会出现变色效果,这就是css字体变色,一方面是为了主动吸引人用户目光去点击,另一方面是为了防止用户点击错其他文字段落。其实这种css鼠标经过字体变色的效果是非常容易实现的。只要你略懂css知…

初识python之 APP store排行榜 蜘蛛抓取(一)

直接上干货!! 采用python 2.7.5-windows 打开 http://www.apple.com/cn/itunes/charts/free-apps/ 如上图可以见采用的是utf-8 编码 经过一番思想斗争 编码如下 (拍砖别打脸) #codingutf-8 import urllib2 import urllib …

PP团队圣经巨著《Application Architecture Guide2.0》14章-数据访问层

第十四章 数据访问层指导 概览 这一章主要描述设计数据访问层时要注意的主要原则。它们覆盖了设计数据访问层遇到的通常问题及错误。下面的图表展示了数据层怎样嵌入一个通用的应用架构。 (cnblog我的图片一直上传不了,报Remote server Error,只能使用网络图片了) 数…

20个Flutter实例视频教程-第03节: 不规则底部工具栏制作-1

第03节: 不规则底部工具栏制作-1 博客地址: https://jspang.com/post/flutterDemo.html#toc-973 视频地址: https://www.bilibili.com/video/av39709290?p3 视频里面的评论:动态组件就是可以setState的组件 flutter create demo02的项目 这里…

python模块之smtplib: 用python发送SSL/TLS安全邮件

转载请注明原文出自 http://blog.csdn.net/zhaoweikid/ python的smtplib提供了一种很方便的途径发送电子邮件。它对smtp协议进行了简单的封装。smtp协议的基本命令包括: HELO 向服务器标识用户身份 MAIL 初始化邮件传输 mail from: RCPT 标识单个的邮件…

B-树

6.7 B-树★4◎3 1.B-树的定义  B-树是一种平衡的多路查找树,它在文件系统中很有用。  定义:一棵m阶的B-树,或者为空树,或为满足下列特性的m叉树:  (1)树中每个结点至多有m棵子…

mysql数据库交叉连接,MySQL数据库联合查询与连接查询

联合查询基本概念联合查询是可合并多个相似的选择查询的结果集。等同于将一个表追加到另一个表,从而实现将两个表的查询组合在一起,使用为此为UNINO或UNION ALL联合查询:将多个查询的结果合并到一起(纵向合并):字段数不变&#xf…

原创:MD5 32位加密软件

网站后台数据库切勿使用明文保存密码,否则一旦黑客拿下你的Webshell,后果不堪设想。网站后台密码加密大多数采用的就是MD5算法加密。今天给大家送一个本人用c#简单编写的MD5 32位加密程序,虽然没有什么技术含量,但保证没有后门。 …

(教学思路 c#之类一)声明类和对象、定义类成员及其引用

上一节(教学思路 c#之面向对象二)初步理解面向对象的基本概念中,我没有提到任何的代码,只是用语言和实例来说明什么是类和对象以及面向对象的特性等基本概念,类是c#程序语言的重要核心,也是构建应用程序最主…

【springboot】之自动配置原理

使用springboot开发web应用是很方便,只需要引入相对应的GAV就可以使用对应的功能,springboot默认会帮我们配置好一些常用配置。那么springboot是怎么做到的呢?这篇文章将一步步跟踪源码,查看springboot到底是如何帮我们做自动化配置。 sprin…

阴雨连绵潮湿加剧 车辆防潮提升保值

近日来,申城阴雨绵绵,不但增加了行车的难度,也使爱车潮气严重,开上一会就会发现前车窗布满水汽,需要开空调吹干才能保证良好视野。此外潮气也容易对人体和车辆本身造成影响,首当其冲的是车内电器&#xff0…

php nsdata,iOS开发之数据存储之NSData

1、概述使用archiveRootObject:toFile:方法可以将一个对象直接写入到一个文件中,但有时候可能想将多个对象写入到同一个文件中,那么就要使用NSData来进行归档对象。NSData可以为一些数据提供临时存储空间,以便随后写入文件,或者存…

asp.net控件开发基础(19)

上两篇讨论了基本数据绑定控件的实现步骤,基本上我们按着步骤来就可以做出简单的数据绑定控件了。过年前在看DataGrid的实现,本来想写这个的,但2.0出了GridView了,再说表格控件实现比较复杂,所以先放着。我们一起打开M…

1048 Find Coins

水题&#xff0c;详见代码&#xff5e; #include <iostream> #include <string.h> #include <cstdio> #include <algorithm> #include <cstdlib> #include <math.h> #include <queue> #include <stack> #include <vector&g…

php组件是啥,浅谈PHP组件、框架以及Composer

本篇文章主要介绍了PHP组件、框架以及Composer&#xff0c;具有一定的学习价值&#xff0c;感兴趣的朋友可以了解一下。什么是组件组件是一组打包的代码&#xff0c;是一系列相关的类、接口和Trait&#xff0c;用于帮助我们解决PHP应用中某个具体问题。例如&#xff0c;你的PHP…

{转}maven+continuum安装与配置

为什么80%的码农都做不了架构师&#xff1f;>>> 一、下载 1. maven: 项目管理工具&#xff0c;可以进行项目的编译、测试、布置、发布等 下载路径&#xff1a;http://apache.etoak.com/maven/binaries/apache-maven-2.2.1-bin.zip 2. continuum: 基本实现的功能&…

.net Excel导出出现乱码及excel打开出现错误提示

测试人员测试发现。导出excel出现乱码以及出现文件可正常导出、后几次导出却异常。 使用的系统浏览器&#xff1a; 其他360浏览器、谷歌浏览器均正常。 出现情况如下&#xff1a; 查资料发现&#xff0c;之前导出代码编写的 Response.AppendHeader("Content-Disposition&q…

rsync推拉模型及结合inotify实现推模型自动同步

一、前言 无论使用什么操作系统下&#xff0c;都经常有同步文件的需求&#xff0c;不管发生在本地&#xff0c;还是发生在本地和远程主机之间。那么应该怎么做呢&#xff1f; 使用拷贝类的命令&#xff0c;本地使用cp命令&#xff0c;复制到远程主机使用scp这样的命令&#xff…