///发送WM_COPYDATA消息的数据结构 /// internal struct COPYDATASTRUCT { /// ///用户自定义数据 /// internal IntPtr dwData; /// ///数据长度 /// internal int cbData; /// ///数据首地址指针 /// internal IntPtr lpData; }现在我们来看看具体如何实现:1、 我们首先创建一个类,让它从Form类继承,因为我们需要一个窗体,然后对它的消息进行处理: internal class WinMsg : Form{ private string _messageString; private List _windowList; private int _message; private int _intHandler; private bool _isConnected;} 2、 然后定义构造函数,在构造函数里注册一个消息通道值,并发出一个广播通知其它在这个通道的窗口。 internal WinMsg(string messageString) { _messageString = messageString; _isConnected = false; _intHandler = Handle.ToInt32(); _windowList = new List(); _message = Win32.RegisterWindowMessage(_messageString);//注册一个消息通道 int errCode = Win32.PostMessage(Win32.HWND_BROADCAST, _message, Win32.CONNECTION, _intHandler);//向此通道内所有的窗口广播自己的句柄 if (errCode == 0) { throw new Win32Exception(errCode);//发生错误,抛出异常 } else { _isConnected = true; } }3、 重写基类的WndProc函数,处理接收到的消息: protected override void WndProc(ref Message m) { if (m.Msg == _message)//接收到广播消息,进行处理 { int LParam = m.LParam.ToInt32(); int WParam = m.WParam.ToInt32(); if (LParam != 0 && LParam != _intHandler) { if (WParam == Win32.DISCONNECTION) { _windowList.Remove(m.WParam);//将对方窗口的句柄从列表中删除 } else { if (WParam==Win32.CONNECTION) { Win32.PostMessage(m.LParam, _message, Win32.REVERSION, _intHandler); } _windowList.Add(m.LParam);//将对方窗口的句柄存入列表中 } } return; } switch (m.Msg) { case Win32.WM_COPYDATA://接收到其它窗口发送过来的数据 { COPYDATASTRUCT data = new COPYDATASTRUCT(); data = (COPYDATASTRUCT)m.GetLParam(data.GetType()); byte[] message = new byte[data.cbData]; Marshal.Copy(data.lpData, message, 0, data.cbData);//从非托管内存中将数据复制到我们的byte数组中。 Anyzler(m.WParam, message);//在这里处理接收到的数据。 } break; case Win32.WM_DESTROY: case Win32.WM_QUERYENDSESSION://窗口被关闭,向其它窗口广播通知从队列中删除自己 Win32.PostMessage(Win32.HWND_BROADCAST, _message, Win32.DISCONNECTION, _intHandler); base.WndProc(ref m);//调用基类的消息处理函数。 break; default: base.WndProc(ref m); //调用基类的消息处理函数。 break; } }4、 定义发送消息的函数: internal void Send(byte[] message) { if (_isConnected) { int length = message.Length; IntPtr ptr = Marshal.AllocHGlobal(length);//申请一块非托管内存 Marshal.Copy(message, 0, ptr, length);//将要发送的数据封送到非托管内存 COPYDATASTRUCT data = new COPYDATASTRUCT(); data.dwData = IntPtr.Zero;//这里可以随意定义。 data.cbData = length;//传递要发送的数据的长度 data.lpData = ptr;//传递要发送的数据的地址指针 //向其它所有窗口发送数据,这里不能发广播消息。必须一个一个发送。 foreach (IntPtr window in _windowList) { Win32.SendMessage(window, Win32.WM_COPYDATA, this.Handle, ref data); } Marshal.FreeHGlobal(ptr);//释放这块非托管内存,这行一定要写上,不然会内存泄漏。 } }好了,主要代码就是这些了!-IMDN开发者社群-imdn.cn">
///发送WM_COPYDATA消息的数据结构 /// internal struct COPYDATASTRUCT { /// ///用户自定义数据 /// internal IntPtr dwData; /// ///数据长度 /// internal int cbData; /// ///数据首地址指针 /// internal IntPtr lpData; }现在我们来看看具体如何实现:1、 我们首先创建一个类,让它从Form类继承,因为我们需要一个窗体,然后对它的消息进行处理: internal class WinMsg : Form{ private string _messageString; private List _windowList; private int _message; private int _intHandler; private bool _isConnected;} 2、 然后定义构造函数,在构造函数里注册一个消息通道值,并发出一个广播通知其它在这个通道的窗口。 internal WinMsg(string messageString) { _messageString = messageString; _isConnected = false; _intHandler = Handle.ToInt32(); _windowList = new List(); _message = Win32.RegisterWindowMessage(_messageString);//注册一个消息通道 int errCode = Win32.PostMessage(Win32.HWND_BROADCAST, _message, Win32.CONNECTION, _intHandler);//向此通道内所有的窗口广播自己的句柄 if (errCode == 0) { throw new Win32Exception(errCode);//发生错误,抛出异常 } else { _isConnected = true; } }3、 重写基类的WndProc函数,处理接收到的消息: protected override void WndProc(ref Message m) { if (m.Msg == _message)//接收到广播消息,进行处理 { int LParam = m.LParam.ToInt32(); int WParam = m.WParam.ToInt32(); if (LParam != 0 && LParam != _intHandler) { if (WParam == Win32.DISCONNECTION) { _windowList.Remove(m.WParam);//将对方窗口的句柄从列表中删除 } else { if (WParam==Win32.CONNECTION) { Win32.PostMessage(m.LParam, _message, Win32.REVERSION, _intHandler); } _windowList.Add(m.LParam);//将对方窗口的句柄存入列表中 } } return; } switch (m.Msg) { case Win32.WM_COPYDATA://接收到其它窗口发送过来的数据 { COPYDATASTRUCT data = new COPYDATASTRUCT(); data = (COPYDATASTRUCT)m.GetLParam(data.GetType()); byte[] message = new byte[data.cbData]; Marshal.Copy(data.lpData, message, 0, data.cbData);//从非托管内存中将数据复制到我们的byte数组中。 Anyzler(m.WParam, message);//在这里处理接收到的数据。 } break; case Win32.WM_DESTROY: case Win32.WM_QUERYENDSESSION://窗口被关闭,向其它窗口广播通知从队列中删除自己 Win32.PostMessage(Win32.HWND_BROADCAST, _message, Win32.DISCONNECTION, _intHandler); base.WndProc(ref m);//调用基类的消息处理函数。 break; default: base.WndProc(ref m); //调用基类的消息处理函数。 break; } }4、 定义发送消息的函数: internal void Send(byte[] message) { if (_isConnected) { int length = message.Length; IntPtr ptr = Marshal.AllocHGlobal(length);//申请一块非托管内存 Marshal.Copy(message, 0, ptr, length);//将要发送的数据封送到非托管内存 COPYDATASTRUCT data = new COPYDATASTRUCT(); data.dwData = IntPtr.Zero;//这里可以随意定义。 data.cbData = length;//传递要发送的数据的长度 data.lpData = ptr;//传递要发送的数据的地址指针 //向其它所有窗口发送数据,这里不能发广播消息。必须一个一个发送。 foreach (IntPtr window in _windowList) { Win32.SendMessage(window, Win32.WM_COPYDATA, this.Handle, ref data); } Marshal.FreeHGlobal(ptr);//释放这块非托管内存,这行一定要写上,不然会内存泄漏。 } }好了,主要代码就是这些了! - IMDN开发者社群-imdn.cn">
首页 »
C# »
C# 不同进程间通讯实例源码下载(SendMessage方式)
于 2013-09-14 发布
0 210
在.NET中基于Windows消息的IPC实现 一、什么是IPC IPC(Inter process Communication)就是“进程间通讯”。我们都知道,在windows系统中,各个应用程序(进程)之间常常需要交换、传递数据,这就要解决进程间的数据通信问题。在最初的16位Windows3.x系统中,所有Windows应用程序共享单一地址,任何进程都能够对这一共享地址空间的数据进行读写操作。 随着Windwos98、Windows NT、Windows2000等32位的操作系统的出现,每个进程都有自己的地址空间,一个Windows进程不能存取另一个进程的私有数据,也就是说,虽然两个进程可以用具有相同值的指针寻址,但所读写的只是它们各自的数据,这样就减少了进程之间的相互干扰。 二、如何实现IPC 那么在windows当前系统下,如何实现进程通讯呢?其实有很多方法,如: 1、 剪贴板Clipboard 2、 DDE(动态数据交换) 3、 内存映像 4、 消息管道 5、 邮件槽 6、 Socket 7、 RPC 8、 串行/并行通信(Serial/Parallel Communication) 9、 COM/DCOM 10、Windows消息 三、基于Windows消息的IPC 现在让我们进入今天我们要讲的主题:“基于Windows消息的IPC实现”。 在这里,我假定大家对Windows消息机制都有很好的理解,所以我就不在这上面费太多的墨水了。我们直接看看Windows消息是怎么样实现进程间通讯的。我们首先看看Windows的消息常数: WM_COPYDATA=0x004A// 当一个应用程序传递数据给另一个应用程序时发送此消息。 这就是我们要的。下面我们来看看如何利用它来实现IPC。 让我们先看看几个API函数,没有它们,我们没有办法将数据发送出去。 1、 PostMessage 函数功能:该函数将一个消息放入(寄送)到与指定窗口创建的线程相联系消息队列里,不等待线程处理消息就返回。消息队列里的消息通过调用GetMessage和PeekMessage取得。 函数原型:B00L PostMessage(HWND hWnd,UINT Msg,WPARAM wParam,LPARAM lParam); 参数 hWnd:其窗口程序接收消息的窗口的句柄。可取有特定含义的两个值: HWND.BROADCAST:消息被寄送到系统的所有顶层窗口,包括无效或不可见的非自身拥有的窗口、被覆盖的窗口和弹出式窗口。消息不被寄送到子窗口。 NULL:此函数的操作和调用参数dwThread设置为当前线程的标识符PostThreadMessage函数一样。 Msg:指定被寄送的消息。 wParam:指定附加的消息特定的信息。 IParam:指定附加的消息特定的信息。 返回值:如果函数调用成功,返回非零值:如果函数调用失败,返回值是零。若想获得更多的错误信息,请调用GetLastError函数。 2、 SendMessage 函数功能:该函数将指定的消息发送到一个或多个窗口。此函数为指定的窗口调用窗口程序,直到窗口程序处理完消息再返回。而函数PostMessage不同,将一个消息寄送到一个线程的消息队列后立即返回。 函数原型:LRESULT SendMessage(HWND hWnd,UINT Msg,WPARAM wParam,LPARAM IParam); 参数: hWnd:其窗口程序将接收消息的窗口的句柄。如果此参数为HWND_BROADCAST,则消息将被发送到系统中所有顶层窗口,包括无效或不可见的非自身拥有的窗口、被覆盖的窗口和弹出式窗口,但消息不被发送到子窗口。 Msg:指定被发送的消息。 wParam:指定附加的消息指定信息。 IParam:指定附加的消息指定信息。 返回值:返回值指定消息处理的结果,依赖于所发送的消息。 3、 RegisterWindowMessage 函数功能:RegisterWindowMessage函数定义一个新的窗口消息,该消息保证在整个系统范围内是唯一的。调用SendMessage或PostMessage函数时可以使用该函数返回的消息值。 函数原型:UINT RegisterWindowMessage(lpsz) 参数: lpsz指向一个以NULL结束的字符串,该字符串指定待登记的消息。 返回值:若成功地登记了消息,返回值是一个消息标识符。该标识符值的范围在0XC000到0XFFFF之间,否则,返回值为0。 我们现在在C#中声明这些API函数: [DllImport("user32")] internal static extern int RegisterWindowMessage(string lpString); [DllImport("user32")] internal static extern int PostMessage(IntPtr hWnd, int Msg, int wParam, int lParam); [DllImport("user32")] internal static extern int PostMessage(IntPtr hwnd, int wMsg, IntPtr wParam, IntPtr lParam); [DllImport("user32")] internal static extern int SendMessage(IntPtr hWnd, int Msg, IntPtr wParam, ref COPYDATASTRUCT lParam); 然后定义一些我们需要的常数: internal const int WM_COPYDATA = 0x004A; //当一个应用程序传递数据给另一个应用程序时发送此消息 internal const int WM_DESTROY = 0x0002; //窗体被销毁 internal const int WM_CREATE = 0x0001; //应用程序创建一个窗口 internal const int WM_QUERYENDSESSION = 0x0011; //当用户选择结束对话框或程序自己调用ExitWindows函数 internal static readonly IntPtr HWND_BROADCAST = new IntPtr(0xFFFF); 我们还需要一个传送数据的结构: /// ///发送WM_COPYDATA消息的数据结构 /// internal struct COPYDATASTRUCT { /// ///用户自定义数据 /// internal IntPtr dwData; /// ///数据长度 /// internal int cbData; /// ///数据首地址指针 /// internal IntPtr lpData; } 现在我们来看看具体如何实现: 1、 我们首先创建一个类,让它从Form类继承,因为我们需要一个窗体,然后对它的消息进行处理: internal class WinMsg : Form{ private string _messageString; private List _windowList; private int _message; private int _intHandler; private bool _isConnected;} 2、 然后定义构造函数,在构造函数里注册一个消息通道值,并发出一个广播通知其它在这个通道的窗口。 internal WinMsg(string messageString) { _messageString = messageString; _isConnected = false; _intHandler = Handle.ToInt32(); _windowList = new List(); _message = Win32.RegisterWindowMessage(_messageString);//注册一个消息通道 int errCode = Win32.PostMessage(Win32.HWND_BROADCAST, _message, Win32.CONNECTION, _intHandler);//向此通道内所有的窗口广播自己的句柄 if (errCode == 0) { throw new Win32Exception(errCode);//发生错误,抛出异常 } else { _isConnected = true; } } 3、 重写基类的WndProc函数,处理接收到的消息: protected override void WndProc(ref Message m) { if (m.Msg == _message)//接收到广播消息,进行处理 { int LParam = m.LParam.ToInt32(); int WParam = m.WParam.ToInt32(); if (LParam != 0 && LParam != _intHandler) { if (WParam == Win32.DISCONNECTION) { _windowList.Remove(m.WParam);//将对方窗口的句柄从列表中删除 } else { if (WParam==Win32.CONNECTION) { Win32.PostMessage(m.LParam, _message, Win32.REVERSION, _intHandler); } _windowList.Add(m.LParam);//将对方窗口的句柄存入列表中 } } return; } switch (m.Msg) { case Win32.WM_COPYDATA://接收到其它窗口发送过来的数据 { COPYDATASTRUCT data = new COPYDATASTRUCT(); data = (COPYDATASTRUCT)m.GetLParam(data.GetType()); byte[] message = new byte[data.cbData]; Marshal.Copy(data.lpData, message, 0, data.cbData);//从非托管内存中将数据复制到我们的byte数组中。 Anyzler(m.WParam, message);//在这里处理接收到的数据。 } break; case Win32.WM_DESTROY: case Win32.WM_QUERYENDSESSION://窗口被关闭,向其它窗口广播通知从队列中删除自己 Win32.PostMessage(Win32.HWND_BROADCAST, _message, Win32.DISCONNECTION, _intHandler); base.WndProc(ref m);//调用基类的消息处理函数。 break; default: base.WndProc(ref m); //调用基类的消息处理函数。 break; } }4、 定义发送消息的函数: internal void Send(byte[] message) { if (_isConnected) { int length = message.Length; IntPtr ptr = Marshal.AllocHGlobal(length);//申请一块非托管内存 Marshal.Copy(message, 0, ptr, length);//将要发送的数据封送到非托管内存 COPYDATASTRUCT data = new COPYDATASTRUCT(); data.dwData = IntPtr.Zero;//这里可以随意定义。 data.cbData = length;//传递要发送的数据的长度 data.lpData = ptr;//传递要发送的数据的地址指针 //向其它所有窗口发送数据,这里不能发广播消息。必须一个一个发送。 foreach (IntPtr window in _windowList) { Win32.SendMessage(window, Win32.WM_COPYDATA, this.Handle, ref data); } Marshal.FreeHGlobal(ptr);//释放这块非托管内存,这行一定要写上,不然会内存泄漏。 } }好了,主要代码就是这些了!
下载说明:请别用迅雷下载,失败请重下,重下不扣分!
InstantMessaging
简单的网络聊天文件传输程序 可以
0.普通聊天(文字、图片)
1.聊天室
2.传输文件(断点续传)
3.共享屏幕
4.白板
5.视频、音频聊天(Simple network chat file transfer program can 0. General chat (text, images) a chatroom 2. Transfer files (HTTP) 3. Shared screen 4. Whiteboard 5 video and audio chat)
2013-08-26 22:14:23下载
积分:1
期望最大化MLE算法(MLE.m)
期望最大化MLE算法(MLE.m)
2021-05-06下载
积分:1
ava-kuangjia
ava 框架,以前找的,不知道现在能用不,需要的自己下载(ava kuangjia )
2013-12-21 21:33:50下载
积分:1
AUpdateProvider
一个完善的升级程序,可供编写自动升级程序模块参考。(A complete update program solution.)
2014-01-18 00:38:42下载
积分:1
C#远程 桌面管理工具源码
C#实现远程桌面连接
2019-02-27下载
积分:1
VC++ 五子棋包括网 络 版与单机版
利用C++语言编写的五子棋单机版和网络版的结合,需要的朋友可以下载,里面附有源码,没有夹杂其他平台的纯VC++(The combination of gobang single edition and network version written in C++ language needs friends to download, which contains source code, no pure VC++ mixed with other platforms.)
2017-12-13 22:19:10下载
积分:1
CSharpremoting
仿照飞信,网络编程。像飞信一样,自由添加客户端用户。(Modeled Fetion, network programming.)
2014-06-16 14:25:11下载
积分:1
VC-OPC-c lient
说明: 基于VC的OPC客户端软件研究与实现.......(The OPC-based VC research and implementation of client software .......
)
2011-04-16 11:52:41下载
积分:1
基于STM32F103ZET6实现的智能灯光亮度调节器
【实例简介】本实验是一个期末大作业,是我写的第一个小项目。按键控制自动识别模式,或者手动调节模式。可调五档光的亮度。也可通过环境光的亮度自动调节,环境越亮,灯越暗。本代码由正点原子精英开发板完成,可直接烧入使用。
2021-08-02 00:31:02下载
积分:1
c # 扫描IP Http Header
c# 扫描IP Http Headerusing System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using System.Drawing;using System.Linq;using System.Text;using System.Windows.Forms;using System.Threading;using System.IO;namespace HScan{ public partial class Form1 : Form { int _currentThreads = 0; int _maxThreads = 100; Thread main = null; Thread mt = null; List threads = new List(); public Form1() { InitializeComponent(); Control.CheckForIllegalCrossThreadCalls = false; } private void btnStart_Click(object sender, EventArgs e) { btnStart.Enabled = false; if (txtStart.Text.Trim() == "") { MessageBox.Show("起始IP不能为空."); return; } if (txtEnd.Text.Trim() == "") { MessageBox.Show("结束IP不能为空."); return; } int ts = Convert.ToInt32(txtThreads.Text); _maxThreads = ts; string startIp = txtStart.Text; string endIp = txtEnd.Text; TParameter tp=new TParameter(); tp.StartIp=startIp; tp.EndIp=endIp; tp.ThreadCount=ts; main = new Thread(new ParameterizedThreadStart(StartMe)); main.Start(tp); } protected void ThreadManage() { Thread c=null; while (true) { System.Object lockThis = new System.Object(); lock (lockThis) { for (int i = 0; i < threads.Count; i ) { if (threads[i] != null && !threads[i].IsAlive) { c = threads[i]; break; } } if (c != null) { threads.Remove(c); } } } } protected void StartMe(object ob) { mt = new Thread(new ThreadStart(ThreadManage)); mt.Start(); TParameter p = ob as TParameter; string curIp = p.StartIp; while (true) { for (int i = 0; i < _maxThreads; i ) { if (curIp != "") { if (_currentThreads >= _maxThreads) break; System.Object lockThis = new System.Object(); lock (lockThis) { _currentThreads ; if (_currentThreads > _maxThreads) _currentThreads = _maxThreads; string tip = curIp; Thread t = new Thread(new ParameterizedThreadStart(Run)); t.Start(tip); threads.Add(t); curIp = IPUtility.getLastIp(curIp, p.EndIp, 1); } } else { break; } } } } protected void Run(object ob) { string ip = ob.ToString(); SocketGetHead h = new SocketGetHead(); string ret = h.GetHtml(ip, 80); if (ret.IndexOf("DVRDVS-Webs") > 0) { ListViewItem item = new ListViewItem(); item.SubItems[0].Text = (listView1.Items.Count 1).ToString(); ListViewItem.ListViewSubItem lvSubItem = new ListViewItem.ListViewSubItem(); lvSubItem.Text = ip; item.SubItems.Add(lvSubItem); lvSubItem = new ListViewItem.ListViewSubItem(); lvSubItem.Text = "DVRDVS-Webs"; item.SubItems.Add(lvSubItem); listView1.Items.Add(item); } System.Object lockThis = new System.Object(); lock(lockThis) { lblCurIp.Text = ip; _currentThreads--; if (_currentThreads < 0) _currentThreads = 0; } } private void tsmCopy_Click(object sender, EventArgs e) { if (listView1.SelectedItems.Count > 0) { string ip = listView1.SelectedItems[0].SubItems[1].Text; Clipboard.SetText(ip); } } private void tsmExport_Click(object sender, EventArgs e) { StreamWriter writer = new StreamWriter(AppDomain.CurrentDomain.BaseDirectory "\export.txt",true); foreach (ListViewItem item in listView1.Items) { string ip=item.SubItems[1].Text; writer.WriteLine(ip); writer.Flush(); } writer.Flush(); writer.Close(); MessageBox.Show("导出成功!"); } private void Form1_FormClosing(object sender, FormClosingEventArgs e) { try { if (mt != null) { mt.Interrupt(); mt.Abort(); } foreach (Thread t in threads) { t.Interrupt(); t.Abort(); } if (main != null) { main.Interrupt(); main.Abort(); } } catch { } Thread.Sleep(5000); } private void btnStop_Click(object sender, EventArgs e) { try { if (mt != null) { mt.Interrupt(); mt.Abort(); } foreach (Thread t in threads) { t.Interrupt(); t.Abort(); } if (main != null) { main.Interrupt(); main.Abort(); } } catch { } btnStart.Enabled = true; } }}
2014-06-23下载
积分:1