搬运自:http://wurang.me/2014/04/24/dotnet-IPC.html
如果有一个需求,用一个程序控制另一个程序,最简单的,比如用程序A打开程序B,这个想必平时都会用到,可以使用Process类及相关的方法。那么再打开B的时候发送一些参数,然后B根据这些参数做出一些反映,这该怎么实现?其实还是用Process。
发送端:
static void Main(string[] args)
{
Console.WriteLine("请输入接收器路径:");
string path = Console.ReadLine();
Console.WriteLine("请输入接收器启动参数:");
string para = Console.ReadLine();
ProcessStartInfo pi = new ProcessStartInfo();
pi.FileName = @path;
pi.Arguments = para;try
{
Process.Start(pi);
}
catch
{
Console.WriteLine("找不到接收器或出现错误!");
}
Console.ReadKey();
}
接收端:
static void Main(string[] args)
{
if (args.Length == 0)
{
Console.WriteLine("未接到信息!");
}
else
{
foreach (string s in args)
{
Console.WriteLine(s);
}
}Console.ReadKey();
}
这样我们就可以用程序A启动程序B,根据A传入的参数,程序B做出相应的处理。不过在WPF中,就没法直接用Main中的args参数了。对于WPF,可以用下面的方式处理:
1.在App.xaml 中删除 StartupUri,并添加Startup
<Application x:Class="WpfApplication65.App"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Startup="Application_Startup" >
<Application.Resources></Application.Resources>
</Application>
2.在App.xaml.cs中填写startup的内容
private void Application_Startup(object sender, StartupEventArgs e)
{
MainWindow mw = new MainWindow();
foreach(string s in e.Args)
{
mw.txtShow.Text += s;
}
mw.Show();
}
这样就可以获取传入的参数了。
但是再改一下需求,我们不仅仅通过程序启动的时候传入参数,而是需要给一个已经启动的程序传入参数,那么就需要用进程通信了IPC了。IPC需要用到Windows
API,下面将介绍一下WPF实现进程间的通信。
1.新建数据的结构体类库
新建一个类库,类库内容如下:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;namespace DataStruct
{
[StructLayout(LayoutKind.Sequential)]
public struct DataStruct
{
public IntPtr dwData;
public int cbData; // 字符串长度
[MarshalAs(UnmanagedType.LPStr)]
public string lpData; // 字符串
}
}
2.新建信息帮助类库
新建一个类库,引用上一步的结构体类库,类库内容如下:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;namespace MessageHelper
{
public class MessageHelper
{
public const int WM_DOWNLOAD_COMPLETED = 0x00AA;
public const int WM_COPYDATA = 0x004A;[DllImport("User32.dll", EntryPoint = "FindWindow")]
public static extern IntPtr FindWindow(string lpClassName, string lpWindowName);[DllImport("User32.dll", EntryPoint = "SendMessage")]
public static extern int SendMessage(IntPtr wnd, int msg, int wP, ref DataStruct.DataStruct cds);
}
}
3.发送端
首先引用前两部的类库
发送有两种方式:
a.通过进程名
var lstProcess = Process.GetProcessesByName(txtProcess.Text);
if (lstProcess.Length > 0)
{
Process proc = lstProcess[0];
DataStruct.DataStruct cds;
cds.dwData = IntPtr.Zero;
cds.lpData = txtMSG.Text;
cds.cbData = System.Text.Encoding.Default.GetBytes(txtMSG.Text).Length + 1;int fromWindowHandler = 0;
MessageHelper.MessageHelper.SendMessage(proc.MainWindowHandle, MessageHelper.MessageHelper.WM_COPYDATA, fromWindowHandler, ref cds);
}
注意:使用这种方法,如果窗体的ShowInTaskbar=false,也就是不在任务栏显示的话,那么是没有办法通过MainWindowHandle获取窗口的。
b.通过窗口名
IntPtr hwnd = MessageHelper.MessageHelper.FindWindow(null, txtTitle.Text);if (hwnd != IntPtr.Zero)
{
DataStruct.DataStruct cds;
cds.dwData = IntPtr.Zero;
cds.lpData = txtMSG.Text;
cds.cbData = System.Text.Encoding.Default.GetBytes(txtMSG.Text).Length + 1;
// 消息来源窗体
int fromWindowHandler = 0;
MessageHelper.MessageHelper.SendMessage(hwnd, MessageHelper.MessageHelper.WM_COPYDATA, fromWindowHandler, ref cds);
}
注意:使用这种方法,如果有多个窗口的Title是一样的,也是会有冲突的。
4.接收端
首先还是先引用前两步的类库。
public MainWindow()
{
InitializeComponent();
Loaded += new RoutedEventHandler(Window_Loaded);
}
private void Window_Loaded(object sender, RoutedEventArgs e)
{
(PresentationSource.FromVisual(this) as HwndSource).AddHook(new HwndSourceHook(this.WndProc));
}IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
{if (msg == MessageHelper.MessageHelper.WM_COPYDATA)
{DataStruct.DataStruct cds = (DataStruct.DataStruct)System.Runtime.InteropServices.Marshal.PtrToStructure(lParam, typeof(DataStruct.DataStruct));
txtShow.Text = cds.lpData;
}
return hwnd;
}
.Net程序间的通讯与控制,码迷,mamicode.com