c# 如何实现不同进程之间的通信

  进程之间的通信是为了解决不同进程之间的数据传输问题,这样可以让不同程序交互数据。实现进程通信的方式:1、剪切板;2、COM;3、内存映射文件;4、WCF

1、剪切板Clipboard在进程间传送对象

  剪切板是一个供应用程序使用的公有区域。在.NET中定一个了一个DataFormats类,此类包含一些静态字段,定义了剪切板中可以存放的数据类型。使用Clipboard类可以向剪切板中放入数据。

  如将文字放入剪切板,使用方法SetDataObject即可:Clipboard.SetDataObject("剪切板文字2"); 在读取的时候,先判断剪切板中是否有文字,然后再读取:

IDataObject data = Clipboard.GetDataObject();
if (data.GetDataPresent(DataFormats.Text))
{
  label1.Text = data.GetData(DataFormats.Text).ToString();
}

  将自定义的数据放置到剪切板,自定义一个图片类,并标记为可序列化(此处使用的命名空间是:TestClipboard)。将自定义数据类型对象放置到剪切板的关键是DataObject类,它实现了IDataObject接口。它就像一个容器,存放将被放置在剪切板上的数据。

[Serializable]
public class MyPic
{
  /// <summary>
  /// 图片
  /// </summary>
  public Image Img;
  /// <summary>
  /// 图片信息
  /// </summary>
  public string ImgInfo;
}
public void SetMyPicToClipboard()
{
  MyPic obj = new MyPic();
  obj.Img = Properties.Resources.Image;
  obj.ImgInfo = "测试将自定义类型保存至剪切板";
  //创建数据对象,并将数据装入
  IDataObject dataObj = new DataObject(obj);

  //其他类型也可以放置在同一数据对象中
  /*
  dataObj.SetData(DataFormats.UnicodeText, "测试文字");
  dataObj.SetData(DataFormats.Bitmap, Properties.Resources.Image);
   */
  //复制到剪切板,第二个参数表示程序退出时不清空
  Clipboard.SetDataObject(dataObj, true);
}

  但是,使用Clipboard.SetDataObject方法将一个DataObject对象放到剪切板后,外界访问时,需要指定对象的完整类型名称。如果某种数据类型只能在指定的进程中访问,则可以使用该方式,指定命名空间。

//首先判断剪切板上是否有我的数据:需要完全限定命名空间类型
 if (Clipboard.ContainsData("WindowsFormsApplication1.MyPic"))
 {
   IDataObject dataObj = Clipboard.GetDataObject();//读取数据
   MyPic myPic = dataObj.GetData("WindowsFormsApplication1.MyPic") as MyPic;//转换数据
   pictureBox1.Image = myPic.Img;
   textBox1.Text = myPic.ImgInfo;
 }

2、使用FileSystemWatcher实现进程同步

  该组件可以监控特定的文件夹或文件,比如在此文件夹中某文件被删除或内容被改变时引发对应的事件。通过该组件让多个进程同时监控一个文件,以此可以充当“临时”进程间通信渠道。
  实现进程同步的关键点是:正确设置文件的共享和读写权限。

/// <summary>
/// 实现写入数据
/// </summary>
/// <param name="fileName"></param>
public void SetText(string fileName)
{
  using (FileStream fs = new FileStream(fileName, FileMode.Create, FileAccess.Write, FileShare.Read))
  {
    using (StreamWriter writer = new StreamWriter(fs, Encoding.UTF8))
    {
      writer.Write("内容");
    }
  }
}
/// <summary>
/// 实现读取数据
/// </summary>
/// <param name="fileName"></param>
public void ReadText(string fileName)
{
  using (FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))
  {
    using (StreamReader reader = new StreamReader(fs, Encoding.UTF8))
    {
      string txt = reader.ReadToEnd();
    }
  }
}

  然后,使用FileSystemWatcher组件的Changed事件监控文件是否发生改变。在网络应用程序中,可以使用此组件监控特定的专用于上传文件的文件夹,当发现用户上传文件之后,系统可以自动启动一系列的处理流程。

3、使用内存映射文件(Memory Mapped File)实现进程通信

  含义:在内存中开辟一块存放数据的专用区域,这区域与硬盘上特定的文件相对应。进程将这块内存区域映射到自己的地址空间中,完成像访问普通内存一样访问它。windows中的系统分页文件和休眠文件就是如此实现的。需要引用命名空间System.IO.MemoryMappedFiles。

  MemoryMappedFile对象表示一个内存映射文件,通过它的CreateFromFile方法根据磁盘现有文件创建内存映射文件(注意,使用完后要立即释放资源,实际上它对应的是操作系统的核心对象)。其中,内存映射的容量在未指定时,默认与文件大小相等。在指定大小时,它的值不能小于文件的现有大小。若指定的大小大于磁盘文件大小,磁盘文件会自动增长到内存映射文件声明的容量大小。

  创建MemoryMappedFile对象后,不能直接对其进行读写,必须使用MemoryMappedViewAccessor对象(内存映射视图访问对象)操作 。可以用MemoryMappedFile对象的方法创建一个访问对象。其中,可以指定需要访问文件的范围,从第几个字节到第几个字节。在写入参数时,也需要指明想哪个位置写入什么。同时也可以使用MemoryMappedViewAccessor的Read方法读取数据。

MemoryMappedFile memoryFile = MemoryMappedFile.CreateFromFile("Text.Config", FileMode.OpenOrCreate, "Config", 1400);//kb;
MemoryMappedViewAccessor accessor = memoryFile.CreateViewAccessor(0, 1024);
accessor.Write(0, '2');

  在同一个进程中,可以针对同一个内存映射文件创建多个“内存映射视图访问对象”,从而允许同时修改同一个文件的不同部分,在关闭这些对象时,由操作系统保证将所有修改都写回原始文件。

  MemoryMappedViewAccessor 的Write和Read有泛型方法,单类型只能是结构体类型(应用类型在程序运行时,计算机无法知道应该向内存映射文件写入多少字节数据,引用类型的对象位于托管堆中,其大小需要经过计算,但非常耗时(而且对象可能引用了其他对象),音效内存映射文件的效率)。
  可以使用序列化方式,将引用对象数据进行序列化后,写入内存映射文件中。

MemoryMappedFile memoryFile = MemoryMappedFile.CreateFromFile("Text.Config", FileMode.OpenOrCreate, "Config", 1400);//kb;
MemoryMappedViewStream stream = memoryFile.CreateViewStream();
MyPic obj = new MyPic();
stream.Seek(0, SeekOrigin.Begin);
new BinaryFormatter().Serialize(stream, obj);

4、使用WCF通过管道实现进程通信

  “管道(Pipe)”是Windows所提供的一种进程间通信机制,用于在两个进程间相互传送数据。Windows提供了两种类型管道:匿名管道(Anonymous Pipe)、命名管道(Named Pipe)

  • 匿名管道:只允许单向通信,由于没有名字,因此要通信的两个进程应该是父子关系,父进程在创建子进程时,负责将代表匿名管道的句柄传送给子进程,子进程可以通过该句柄获取父进程传输的数据。其优点是占用资源少、效率高;缺点是通信进程必须为父子关系,限制了使用场景。
  • 命名管道:这种类型的管道拥有一个在本机唯一的名字,可以用于在一个服务进程和多个客户进程之间进行单/双向通信。命名管道是基于消息的通信模式,即一个进程一次可以向另一方进程连续发生多个消息(消息之间通过消息的定界符进行划分),接收方通过定界符提取完整的消息。

  在命名空间System.IO.Pipes中,提供了一些用于实现基于管道的进程间通信,如AnonymousPipeClientStream和AnonymousPipeServerStream可用于实现匿名管道,而NamedPipeClientStream和NamedPipeServerStream可以实现命名管道。但相对于WCF,其比较繁琐,WCF的管道进程通信更加简便和灵活。

  WCF应用程序使用命名管道实现进程通信:WCF提供了一个NetNamedPipeBinding绑定,它可以在地层使用命名管道实现进程通信。

以上就是c# 如何实现不同进程之间的通信的详细内容,更多关于c# 进程通信的资料请关注我们其它相关文章!

(0)

相关推荐

  • C#串口通信模块使用方法示例

    C#串口模块的使用.使用VS .net框架下WinForm程序应用开发. C#开发的串口通信小工具. 相比于QT添加的串口类,WinForm是通过组件的形式将串口加入到程序中.. 在创建完windows窗体之后,添加组件类,就可以将串口加入到其中. 然后就需要写数据成员,初始化和方法,实现串口的读写功能. 串口类数据成员 SerialPort SComm; // 使用构造函数取串口控件 TextBox MsgRc; // 接收数据成员 //构造函数初始化 public SerialComm(Se

  • C#使用SendMessage实现进程间通信的方法

    本文实例讲述了C#使用SendMessage实现进程间通信的方法.分享给大家供大家参考.具体分析如下: 为了深入理解消息机制,先来做一个测试项目 在新建项目的Form1的代码中,加入方法: protected override void DefWndProc(ref Message m) { if (m.Msg == 0x200) { MessageBox.Show("捕捉到消息"); } else { } base.DefWndProc(ref m); } 此方法重写了窗体的消息截获

  • c# 进程之间的线程同步

    Mutex类.Event类.SemaphoreSlim类和ReaderWriterLockSlim类等提供了多个进程之间的线程同步.  1.WaitHandle 基类 WaitHandle抽象类,用于等待一个信号的设置.可以根据其派生类的不同,等待不同的信号.异步委托的BeginInvoke()方法返回一个实现了IAsycResult接口的对象.使用IAsycResult接口可以用AsycWaitHandle属性访问WaitHandle基类.在调用WaitOne()方法时,线程会等待接收一个和等

  • 详解C# 网络编程系列:实现类似QQ的即时通信程序

    引言: 前面专题中介绍了UDP.TCP和P2P编程,并且通过一些小的示例来让大家更好的理解它们的工作原理以及怎样.Net类库去实现它们的.为了让大家更好的理解我们平常中常见的软件QQ的工作原理,所以在本专题中将利用前面专题介绍的知识来实现一个类似QQ的聊天程序.  一.即时通信系统 在我们的生活中经常使用即时通信的软件,我们经常接触到的有:QQ.阿里旺旺.MSN等等.这些都是属于即时通信(Instant Messenger,IM)软件,IM是指所有能够即时发送和接收互联网消息的软件. 在前面专题

  • 利用C#守护Python进程的方法

    背景# 目前我主要负责的一个项目是一个 C/S 架构的客户端开发,前端主要是通过 WPF 相关技术来实现,后端是通过 Python 来实现,前后端的数据通信则是通过 MQ 的方式来进行处理.由于 Python 进程是需要依赖客户端进程来运行,为了保证后端业务进程的稳定性,就需要通过一个 守护进程 来守护 Python 进程,防止其由于未知原因而出现进程退出的情况.这里简单记录一下我的一种实现方式. 实现# 对于我们的系统而言,我们的 Python 进程只允许存在一个,因此,对应的服务类型要采用单

  • C# NetRemoting实现双向通信

    闲来无事想玩玩双向通信,实现类似QQ的互发消息的功能.于是乎开始学习.Net Remoting. .Net Remoting 是由客户端通过Remoting,访问通道以获得服务端对象,再通过代理解析为客户端对象来实现通信的.也就是说对象是由服务端创建的. 先上代码 首先是ICommand库 using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace ICommand

  • 详解C#多线程编程之进程与线程

    一. 进程 简单来说,进程是对资源的抽象,是资源的容器,在传统操作系统中,进程是资源分配的基本单位,而且是执行的基本单位,进程支持并发执行,因为每个进程有独立的数据,独立的堆栈空间.一个程序想要并发执行,开多个进程即可. Q1:在单核下,进程之间如何同时执行? 首先要区分两个概念--并发和并行 并发:并发是指在一段微小的时间段中,有多个程序代码段被CPU执行,宏观上表现出来就是多个程序能"同时"执行. 并行:并行是指在一个时间点,有多个程序段代码被CPU执行,它才是真正的同时执行. 所

  • C# 获取进程退出代码的实现示例

    我需要写一个程序,让这个程序知道另一个程序是否正常退出,于是就需要获取这个进程的退出代码 在程序如果需要手动退出,可以设置当前的退出代码 static void Main(string[] args) { Environment.Exit(-100); } 这时的程序运行就退出,同时退出的代码就是 -100 这和 C 语言的在 main 函数返回值一样 在 C# 如果想要实现 C 语言的 main 函数的返回值,是通过调用 Environment.Exit 方法 那么其他程序如何拿到这个程序的退

  • C# websocket及时通信协议的实现方法示例

    传统"长轮询"实现Web端即时通讯的问题 WebSocket出现之前,Web端为了实现即时通讯,所用的技术都是Ajax轮询(polling).轮询是在特定的的时间间隔(如每1秒),由浏览器对服务器发出HTTP request,然后由服务器返回最新的数据给客服端的浏览器.这种传统的HTTP request 的模式带来很明显的缺点 – 浏览器需要不断的向服务器发出请求,然而HTTP request 的header是非常长的,里面包含的数据可能只是一个很小的值,这样会占用很多的带宽. 而比较

  • 如何利用c#实现通用守护进程

    1. 下载 源码下载:http://files.cnblogs.com/tianzhiliang/CocoWatcher.rar 安装包下载:http://files.cnblogs.com/tianzhiliang/CocoWatcher_Setup.rar 本地下载:http://xiazai.jb51.net/201910/yuanma/CocoWatcher(jb51net).rar 2. 安装注意事项 在配置档中配置你要守护的应用程序,应用程序之间用逗号隔开: <?xml versio

随机推荐