C#自定义消息通信往往采用事件驱动的方式实现,但有时候我们不得不采用操作系统的消息通信机制,例如在和底层语言开发的DLL交互时,是比较方便的。下面列举了一些实现方式,供大家参考:一、通过SendMessage或postmessage函数发送:

1、 定义消息
在C++中引用底层的函数很简单,自定义消息如下
#define WM_TEST WM_USER + 101
而在c#中消息需要定义成windows系统中的原始的16进制数字,比如自定义消息
public const int USER = 0x0400;

public const int WM_TEST =USER+101;

2、 发送消息
消息发送是通过windows提供的API函数SendMessage或postmessage来实现的,它的原型定义:

[DllImport("User32.dll",EntryPoint="SendMessage")]
private static extern int SendMessage(
IntPtr hWnd, // 窗体句柄
uint Msg, // 消息的标识符
uint wParam, // 具体取决于消息
uint lParam // 具体取决于消息
);

[DllImport("User32.dll",EntryPoint="PostMessage")]
private static extern int SendMessage(
IntPtr hWnd, // 接收消息的那个窗口的句柄。如设为HWND_BROADCAST,表示投递给系统中的所有顶级窗口。如设为零,表示投递一条线程消息(可参考PostThreadMessage)
uint Msg, // 消息的标识符
uint wParam, // 具体取决于消息
uint lParam // 具体取决于消息
);

至于两个函数的区别这里就不累述了,有兴趣的朋友可以自己查阅资料。

3、 消息接收
消息发出之后,在Form中如何接收呢?我们可以重载DefWinproc函数来接收消息。
protected override void DefWndProc ( ref System.Windows.Forms.Message m )
{
switch(m.Msg)
{
case Message.WM_TEST: //处理消息
break;
default:
base.DefWndProc(ref m); //调用基类函数处理非自定义消息。
break;
}
}

二、使用PostThreadMessage函数向线程发送消息

1、映射消息结构体原型和自定义消息

public struct tagMSG
{
public int hwnd;
public uint message;
public int wParam;
public long lParam;
public uint time;
public int pt;
}

public const int WM_CX_NULL = 0x400 + 100;

2、发送消息

[DllImport("user32.dll")]

private static extern int PostThreadMessage(

                  int threadId, //线程标识

                  uint msg, //消息标识符

                  int wParam, //具体由消息决定

                  int lParam); //具体由消息决定

此函数获取当前线程一个唯一的线程标识符,这点需要特别注意:Win32 API无法识别管理线程,你必须发送消息到Windows的线程ID上,而不是管理线程的ID上。

[DllImport("kernel32.dll")]

private static extern int GetCurrentThreadId();

因此发送消息过程如下:

private int _NewThreadId =GetCurrentThreadId();

PostThreadMessage(_NewThreadId, WM_CX_NULL, 1, 1);

3、接收消息

该函数从调用线程的消息队列里取得一个消息并将其放于指定的结构。此函数可取得与指定窗口联系的消息和由PostThreadMesssge寄送的线程消息。此函数接收一定范围的消息值。GetMessage不接收属于其他线程或应用程序的消息

[DllImport("user32.dll")]
private static extern int GetMessage(

                  ref tagMSG lpMsg, //指向MSG结构的指针,该结构从线程的消息队列里接收消息信息;

                  int hwnd, //取得其消息的窗口的句柄。这是一个有特殊含义的值(NULL)。GetMessage为任何属于调用线程的窗口检索消息;                                              int wMsgFilterMin, //指定被检索的最小消息值的整数

                  int wMsgFilterMax); //指定被检索的最大消息值的整数

接收实现如下:

public void ThreadExectue()
{
_NewThreadId= GetCurrentThreadId(); //发送线程和接收线程一定要是同一个线程,否则接收不到消息
while (GetMessage(ref msg, 0, 0, 0) > 0)
{
if (msg.message == WM_CX_NULL)
{
MessageBox.Show("消息收到!");
}
}
}

三、使用Application.AddMessageFilter拦截系统消息

1、实现消息过滤器接口

internal class MyMessager : IMessageFilter
{
//截取消息,进行处理
public bool PreFilterMessage(ref System.Windows.Forms.Message m)
{
switch (m.Msg)
{
case CUSTOM_MESSAGE:      //拦截自定义消息 
MessageBox.Show("消息收到!");
return true;    
default:
return false;     //返回false则消息未被裁取,系统会处理
}
}
}
2、安装消息过滤器
private void Form1_Load(object sender, EventArgs e)
{
Application.AddMessageFilter(new MyMessager());
}

实例代码:

C#简单实现自定义消息的发送和接收 收藏
//=================================发送窗口代码=============================

using System;

using System.Collections.Generic;

using System.ComponentModel;

using System.Data;

using System.Drawing;

using System.Text;

using System.Windows.Forms;

/*

制作人:林龙江

制作时间:2007年5月1日

只供参考,有错误之处请指出 !

*/

//手动加入的命名空间

using System.Runtime.InteropServices;

namespace SendCustomMessage

{

public partial class SendForm. Form

{

public SendForm(IntPtr Handle)

{

SendToHandle = Handle;

InitializeComponent();

}

private IntPtr SendToHandle;//这个变量用于保存要发送窗口的句柄

//自定义的消息

public const int USER = 0x500;

public const int MYMESSAGE=USER + 1;

//消息发送API

[DllImport("User32.dll", EntryPoint = "SendMessage")]

private static extern int SendMessage(

IntPtr hWnd, // 信息发住的窗口的句柄

int Msg, // 消息ID

int wParam, // 参数1

ref SENDDATASTRUCT lParam // 参数2 [MarshalAs(UnmanagedType.LPTStr)]StringBuilder lParam

);

//发关按钮

private void Send_Click(object sender, EventArgs e)

{

string myText = textBox1.Text;

byte[] myInfo = System.Text.Encoding.Default.GetBytes(myText);

int len = myInfo.Length;

SENDDATASTRUCT myData;

myData.dwData = (IntPtr)100;

myData.lpData = myText;

myData.DataLength = len + 1;

SendMessage(SendToHandle, MYMESSAGE, 100, ref myData);//发送自定义消息给句柄为SendToHandle 的窗口,

//本例为创建本窗口的窗口句,创建时,传递给本窗口的构造函数

}

}

//要发信息数据结构,作为SendMessage函数的LParam参数

public struct SENDDATASTRUCT

{

public IntPtr dwData; //附加一些个人自定义标志信息,自己喜欢

public int DataLength; //信息的长度

[MarshalAs(UnmanagedType.LPStr)]

public string lpData; //要发送的信息

}

}

//=============================接收窗口代码====================================

using System;

using System.Collections.Generic;

using System.ComponentModel;

using System.Data;

using System.Drawing;

using System.Text;

using System.Windows.Forms;

/*

制作人:林龙江

制作时间:2007年5月1日

只供参考,有错误之处请指出 !

*/

//手动加入的命名空间

using System.Runtime.InteropServices;

namespace SendCustomMessage

{

public partial class Form1 : Form

{

public Form1()

{

InitializeComponent();

sendForm. = new SendForm(this.Handle);

sendForm.Show();

}

SendCustomMessage.SendForm. sendForm;

//自定义消息

public const int USER = 0x500;

public const int MYMESSAGE = USER + 1;

///重写窗体的消息处理函数DefWndProc,从中加入自己定义消息 MYMESSAGE 的检测的处理入口

protected override void DefWndProc(ref Message m)

{

switch (m.Msg)

{

//接收自定义消息MYMESSAGE,并显示其参数

case MYMESSAGE:

SendCustomMessage.SENDDATASTRUCT myData = new SendCustomMessage.SENDDATASTRUCT();//这是创建自定义信息的结构

Type mytype = myData.GetType();

myData = (SendCustomMessage.SENDDATASTRUCT)m.GetLParam(mytype);//这里获取的就是作为LParam参数发送来的信息的结构

textBox1.Text = myData.lpData; //显示收到的自定义信息

break;

default:

base.DefWndProc(ref m);

break;

}

}

}

}

c# 如何处理自定义消息的更多相关文章

  1. ASP.NET Core应用针对静态文件请求的处理[3]: StaticFileMiddleware中间件如何处理针对文件请求

    我们通过<以Web的形式发布静态文件>和<条件请求与区间请求>中的实例演示,以及上面针对条件请求和区间请求的介绍,从提供的功能和特性的角度对这个名为StaticFileMidd ...

  2. Nginx如何处理一个请求

    看了下nginx的官方文档,其中nginx如何处理一个请求讲解的很好,现在贴出来分享下.Nginx首先选定由哪一个虚拟主机来处理请求.让我们从一个简单的配置(其中全部3个虚拟主机都在端口*:80上监听 ...

  3. ASP.NET Core管道深度剖析(3):管道是如何处理HTTP请求的?

    我们知道ASP.NET Core请求处理管道由一个服务器和一组有序的中间件组成,所以从总体设计来讲是非常简单的,但是就具体的实现来说,由于其中涉及很多对象的交互,我想很少人能够地把它弄清楚.为了让读者 ...

  4. 【WP8.1】类似“IT之家” 自定义消息 的实现

    曾经在WP7.WP8下的消息 使用的都是Coding4Fun.Phone.Toolkit里面的ToastPrompt类来实现的. 现在我们来自己做个类似IT之家的这种效果:从右边弹出,经过几秒后会自动 ...

  5. ENode框架Conference案例分析系列之 - 事件溯源如何处理重构问题

    前言 本文可能对大多数不太了解ENode的朋友来说,理解起来比较费劲,这篇文章主要讲思路,而不是一上来就讲结果.我写文章,总是希望能把自己的思考过程尽量能表达出来,能让大家知道每一个设计背后的思考的东 ...

  6. MFC用户自定义消息

    之前做过佳能相机和位移平台的额二次开发,当时遇到一个棘手的问题,就是位移平台如何知道相机已经拍完照了,或者相机如何知道位移平台已经运行到指定位置,当时为了方便采用了定时器来定时检测位移平台的位置,结果 ...

  7. postgresql是如何处理死连接(转)

    在数据库postgresql中,一个客户端到服务器连接实际上是一个tcp socket连接,tcp连接是虚连接,一方非正常退出(如断电),另一方会继续维持这个连接.   举个例子,一个客户端电脑正常连 ...

  8. (转)如何处理iOS中照片的方向

    如何处理iOS中照片的方向 31 May 2015 • 7 min. read • Comments 使用过iPhone或者iPad的朋友在拍照时不知是否遇到过这样的问题,将设备中的照片导出到Wind ...

  9. 新增资产时YTD折旧与累计折旧录入错误如何处理

    如新增资产时YTD折旧与累计折旧录入错误,但资产已入账处理,如何处理: 1.需要先报废资产: 2.需要在总账手工帐冲销未冲抵凭证: 3.重新增加资产,录入资产时YTD折旧及累计折旧金额应为0.  

随机推荐

  1. 得到内网域管理员的5种常见方法<转>

    1.Netbios and LLMNR Name Poisoning 这个方法在WIN工作组下渗透很有用,WIN的请求查询顺序是下面三个步骤:本地hosts文件(%windir%\System32\d ...

  2. 【转】C++ static关键字

    原文:http://blog.csdn.net/hackbuteer1/article/details/7487694 C++的static有两种用法:面向过程程序设计中的static和面向对象程序设 ...

  3. android常见错误--Unable to resolve target ‘android - 8’

    这是由于项目的android的版本没有设置好,解决方法如下: 1,clean项目 选择[project]-[clean] 选中需要进行clean的项目,点击[ok] 2,重新build 选择[proj ...

  4. BZOJ 3505: [Cqoi2014]数三角形 数学

    3505: [Cqoi2014]数三角形 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...

  5. Codeforces Gym 100523C C - Will It Stop? 水题

    C - Will It Stop?Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hust.edu.cn/vjudge/contest/ ...

  6. delphi 动态建立WebBrower

    //Delphi动态建立WebBrowerunit Main;interfaceuses  Windows, Messages, SysUtils, Variants, Classes, Graphi ...

  7. Missing access checks in put_user/get_user kernel API (CVE-2013-6282)

    /* 本文章由 莫灰灰 编写,转载请注明出处. 作者:莫灰灰    邮箱: minzhenfei@163.com */ 1.漏洞成因 Linux kernel对ARM上的get_user/put_us ...

  8. 解析“extern”

    解析“extern” 1. 声明外部变量 现代编译器一般採用按文件编译的方式,因此在编译时,各个文件里定义的全局变量是 互相透明的,也就是说,在编译时,全局变量的可见域限制在文件内部.以下举一个简单的 ...

  9. mysqldump原理1

  10. Linux 学习笔记 文件权限

    * Linux系统会为各种各样的功能创建不同的用户账户,而这些账户并不是真的用户.这些账户称作系统账户,是系统上运行的各种服务进程访问资源用的特殊账户. 所有运行在后台的服务都需要用一个系统用户账户登 ...