在C#中采用的是事件驱动方式,但在我们使用的过程中,有时候通过调用系统原有的消息,处理起来会比较简单一些,特别是在处理与DLL文件的交互时,的确是非常的方便。
   在C#中使用自定义消息
     在C#中使用自定义消息非常简单,只需要下面几个简单的步骤就可以了:
1、  定义消息
定义消息的方法和VC中定义消息有一点点不同
比如在VC中申明一个自定义消息:
#define WM_TEST WM_USER + 101
而在c#中消息需要定义成windows系统中的原始的16进制数字,比如自定义消息
public const int USER = 0x0400;
那么我们在VC中申明的自定义消息,在C#中就可以做对应的声明:
         public const int WM_TEST = USER+101;
2、  发送消息
消息发送是通过windows提供的API函数SendMessage来实现的,它的原型定义:        [DllImport("User32.dll",EntryPoint="SendMessage")]
          private static extern int SendMessage(
            IntPtr hWnd,      // handle to destination window
            uint Msg,         // message
            uint wParam,      // first message parameter
            uint lParam       // second message parameter
       );
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;
            }
}
在C#中使用系统消息

们以WM_PAINT消息的处理为例,在C#中处理消息与MFC的消息处理是类似的,但更为简单。MFC中需要使用
DECLARE_MESSAGE_MAP来定义消息映射,在C#就不需要了。比如WM_PAINT消息,我们只要重载父类中的OnPaint虚拟方法即
可,方法如下:
在菜单View->Other Windows->Object Browser打开对象浏览窗口(或用CTRL+ALT+J打开),在我们的工程名下找到Form并选中,这时在右边的窗口列出所有Form类的成员函数,如图所示:


们选中OnPaint(System.WinForms.PaintEventArgs)此时在下面会显示完整的OnPaint函数protected
void OnPaint ( System.WinForms.PaintEventArgs e
)我们将这一行字符串Copy下来。打开Form1.cs进行代码编辑,我们把刚才拷贝下来的函数定义复制到Form1类里面,并加上override关
键字,此时我们便可以在里面添加我们的消息处理代码了,请参考如下代码段:
protected override void OnPaint (System.Windows.Forms.PaintEventArgs e  )
{
     Font font = new Font("黑体",28);///定义字体:黑体,大小:28
     SolidBrush bluepen = new SolidBrush(Color.Blue);///创建蓝色画笔
     SolidBrush blackpen = new SolidBrush(Color.FromARGB(0xa0,0xa0,0xb0));///创建黑色画笔
     e.Graphics.DrawString("VC知识库",font,blackpen,65,25);///写字符串
     e.Graphics.DrawString("VC知识库",font,bluepen,61,21);///偏移4个象素用不同的颜色再写一次,达到立体效果
}
示例应用
1、  定义消息

们在工程中添加一个Message类用来定义消息。然后添加了三个成员变量,其中USER为自定义消息的初始值,相当与MFC中的WM_USER。
WM_TEST为自定义的用来响应应用程序的消息,WM_MSG为自定义的用来响应DLL传递过来的消息。如何在DLL定义消息请参考文章:VC.Net
从DLL传递消息到DLL。
public class Message
    {
     public const int USER = 0x0400;
//as mfc Define WM_TEST WM_USER + 101
     public const int WM_TEST = USER+101;
     public const int WM_MSG = USER+102;
    }
2、  声明引用函数
在使用消息的地方,申明引用的函数,我们这里在MsgForm.cs文件中申明:
//申明发送消息函数
        [DllImport("User32.dll",EntryPoint="SendMessage")]
        private static extern int SendMessage(
            IntPtr hWnd,      // handle to destination window
            uint Msg,         // message
            uint wParam,      // first message parameter
            uint lParam       // second message parameter
            );

//申明DLL中启动消息函数
        [DllImport("MessageDLL.dll",EntryPoint="StartSendMessage")]
private extern static void StartSendMessage(IntPtr hWnd);

3、  处理系统消息
protected override void OnPaint (  System.Windows.Forms.PaintEventArgs e )
         {
///定义字体:黑体,大小:28
              Font font = new Font("黑体",28);
///创建蓝色画笔
              SolidBrush bluepen = new SolidBrush(Color.Blue);
///创建黑色画笔            
SolidBrush blackpen = new SolidBrush(Color.FromArgb(0xa0,0xa0,0xb0));
///写字符串
e.Graphics.DrawString("VC知识库",font,blackpen,65,25);
///偏移4个象素用不同的颜色再写一次,达到立体效果
             e.Graphics.DrawString("VC知识库",font,bluepen,61,21);       
}
4、   触发自定义消息

//测试应用程序消息
         private void TestAppbutton_Click(object sender, System.EventArgs e)
         {
            SendMessage(this.Handle,Message.WM_TEST,100,200);
         }
        //测试DLL消息
         private void TestDLLbutton_Click(object sender, System.EventArgs e)
         {
            StartSendMessage(this.Handle);
}

5、  响应和处理自定义消息
protected override void DefWndProc ( ref System.Windows.Forms.Message m )
          {
            string message;
            switch(m.Msg)
            {
                case Message.WM_TEST://处理消息
                        message = string.Format("收到从应用程序发出的消息!参数为:{0},{1}",m.WParam,m.LParam);
                    MessageBox.Show(message);///显示一个消息框
                    break;
                case Message.WM_MSG:
                    message = string.Format("收到从DLL发出的消息!参数
为:{0},{1}",m.WParam,m.LParam);
                    MessageBox.Show(message);///显示一个消息框
                    break;
                default:
                    base.DefWndProc(ref m);//调用基类函数处理非自定义消息。

break;
             }
         }
  程序运行结果:
当我们点击测试DLL消息时,弹出消息框显示收到消息的参数,窗口也会调用WM_PAIN函数对窗口进行重新绘制。

C#中自定义消息,与MFc对比的更多相关文章

  1. win32程序中简单应用mfc

    今日写程序在win32中用CRect发现报错,突然想起来.要引入mfc库.想重新建立一个工程添加对mfc的支持.发现选项不能选.查资料后发现. 在win32程序中简单应用mfc库,只需要简单的引入&l ...

  2. Java中两个List对比的算法

    Java中两个List对比的算法:   // 测试数据 // tdcsDdt.add("Z"); // tdcsDdt.add("B"); // tdcsDdt ...

  3. Systemd初始化进程/RHEL 6系统中System V init命令与RHEL 7系统中systemctl命令的对比

    Linux操作系统的开机过程是这样的,即从BIOS开始,然后进入Boot Loader,再加载系统内核,然后内核进行初始化,最后启动初始化进程.初始化进程作为Linux系统的第一个进程,它需要完成Li ...

  4. Linux、Windows 和 Mac 中的换行符对比

    原文地址:Linux.Windows 和 Mac 中的换行符对比 博客地址:http://www.moonxy.com 一.前言 经常使用 Window.Linux 等不同操作系统的开发者,在处理文本 ...

  5. MFC中 自定义消息

    想在对话框显示出来后立即执行一段代码. 方法之一是自定义消息,即添加一个自定义的消息在消息队列中等待对话框初始化之后从消息队列中读取消息执行代码. 在OnInitDialog返回之前post一个自定义 ...

  6. MFC中自定义消息

    在头文件stdafx.h中增加一个自定义消息宏 #define WM_USER_THREADEND WM_USER + 1 在于增加新消息的窗口或对话框类的头文件中增加一个回调函数声明 afx_msg ...

  7. [目录][总结] C++和Java 中的主要操作对比

    总结一些,C++ 和Java 中的一些常用操作对比,就当是自己的查询工具书啦.(暂时按随笔的更新时间排序) [Stack] c++ V.S. Java (2015.04.27) [Map]   c++ ...

  8. VS2008中 ATL CLR MFC Win32 区别

    ATL用于编写COM程序,CLR是.NET的公共语言运行库,MFC是指MFC类库,MFC程序是用这些类库做出的程序,WIN32常规就是不用MFC,使用API函数编的程序.MFC.ATL和CLR是VC2 ...

  9. DELPHI中自定义消息的发送和接收

    DELPHI中的消息处理机制 Delphi是Borland公司提供的一种全新的WINDOWS编程开发工具.由于它采用了具有弹性的和可重用的面向对象Pascal(object-orientedpasca ...

随机推荐

  1. Amazium源码分析:(1)基本介绍

    前言 Amazium是一个网格系统的框架,分析该源码的目的是了解网格系统的实现. 网格系统 定义:设计美观页面布局的方式,上图能够很直观的了解什么是网格系统. 基本概念 column: 列. gutt ...

  2. scala学习笔记(9):Scala函数(2)

    1 指令式编程&函数式编程 指令式:imperative 风格编程.指令式风格,是你常常使用像 Java,C++和 C 这些语言里用的风格,一次性发出一个指令式的命令,用循环去枚举,并经常改变 ...

  3. scala学习笔记(4):占位符

    scala 中占位符的用法 1.作为“通配符”,类似Java中的*.如import scala.math._ 2.:_*作为一个整体,告诉编译器你希望将某个参数当作参数序列处理!例如val s = s ...

  4. HDU 3078 Network

    简单的  RMQ:  先预处理得到  所有 节点的 公共祖先  和  dfs 得到所有节点的父亲节点:  然后  询问时,从自己出发向上找父亲, 然后  得到所有的节点:排序一下 不知道  这题这样也 ...

  5. codeforce Codeforces Round #201 (Div. 2)

    cf 上的一道好题:  首先发现能生成所有数字-N 判断奇偶 就行了,但想不出来,如何生成所有数字,解题报告 说是  所有数字的中最大的那个数/所有数字的最小公倍数,好像有道理:纪念纪念: #incl ...

  6. Android堆栈分析

    在开发中,与界面跳转联系比较紧密的概念是Task(任务)和Back Stack(回退栈).activity的启动模式会影响Task和Back Stack的状态, 进而影响用户体验.除了启动模式之外,I ...

  7. 线上redis服务内存异常分析。

    项目中,新增了一个统计功能,用来统计不同手机型号的每天访问pv,看了下redis2.6有个setbit的功能,于是打算尝尝鲜把 redis从2.4更新到了2.6 因为是租了vps.服务器的内存只有4g ...

  8. @Component @Repository @Service @Controller

    Spring 2.5 中除了提供 @Component 注释外,还定义了几个拥有特殊语义的注释,它们分别是:@Repository.@Service 和 @Controller.在目前的 Spring ...

  9. GC Buffer Busy Waits处理(转载)

    与单实例不同,在RAC环境中,由于多节点的原因,会因为节点间的资源争用产生GC类的等待,而这其中,GC Buffer Busy Waits又是最为常见的,从性能角度上说,RAC是把双刃剑,用的好,能够 ...

  10. a different object with the same identifier value was already associated with **(ssh异常转)

      Hibernate:a different object with the same identifier value was already associated with ...异常解决 今天 ...