c#委托与事件 心得 
c#用委托来实现事件通知机制。委托相当与c++函数指针。整个过程涉及一个呼叫者,一个被呼叫者,还有就是这个委托。

- 实现步骤
有以下几步: 1. 申明委托, 2.定义呼叫者和调用的函数, 3.定义被呼叫者和具体实现的函数(被调用的函数)

1.申明委托 ,在包里或者类里,public
 public delegate void PlayGame(Object sender, EventArgs e); 
2.定义呼叫者(类LetsGame)和调用委托的函数,在呼叫者里要有委托的实例(呼叫者扔出一个委托,被呼叫者给这个委托赋值)
class LetsGame{ 
    public event PlayGame theGame; 
    public void startPlay(EventArgs e){ 
        if(theGame != null){ 
        theGame(this,e); 
    } 
}

3. 定义被呼叫者(类MS)和具体实现的函数(被调用的函数),也就是concrete class的实现或者叫函数指针实例。打个比方,在一个叫MS的类中实现.MS中对呼叫者中委托的实例进行赋值.
class MS {
    public MS(LetsGame lg) {
        lg.theGame += new PlayGame(MSPlayGame);
    }
    public void MSPlayGame(Object sender, EventArgs e){ 
        Console.WriteLine("Who laughs the last who wins"); 
    } 
}

这样当调用LetsGame.startPlay的时候就会调用MS.MSPlayGame.

- 实际应用
对照一下c#的GUI事件处理或者asp.net的web控件事件处理,能帮我们更好的理解委托和事件.大家一定很熟悉asp.net里下面的代码
private void InitializeComponent()
{    
    
    this.Button1.Click += new System.EventHandler(this.Button1_Click);
    
}

private void Button1_Click(object sender, System.EventArgs e)
{
    //do sth
}

这就是用委托来实现事件.你可能发现我们并没有给它声明委托对象并通过event关键字来引用该委托对象,那是因为asp.net早就帮我们做好了该项工作,其委托对象是System.EventHandler. Button1相当于上面的LetsGame的实例,是呼叫者,Button1_Click是被呼叫方法.当你click Button1后,Button1就会调用Button1_Click.

-杂项
我觉得这种机制和design pattern里的observer很类似,我们完全可以用observer来达到同样的效果,但是用委托更灵活,不需要定义一个interface然后所有的concrete class都实现某个方法,函数指针(委托)更灵活.

还有,委托不一定非要和事件一起用,单独用的时候就是函数指针.

-------------------------------------------------------------------------------
WndProc(ref Message m)
protected override void WndProc(ref Message m) 

    const int WM_SYSCOMMAND = 0x0112; 
    const int SC_CLOSE = 0xF060; 
    if (m.Msg == WM_SYSCOMMAND && (int) m.WParam == SC_CLOSE) 
    { 
        // 屏蔽传入的消息事件 
        this.WindowState = FormWindowState.Minimized; 
        return; 
     } 
    base.WndProc(ref m); 
}

protected override void WndProc(ref    Message m)
{
     const int WM_SYSCOMMAND = 0x0112;
     const int SC_CLOSE = 0xF060;
     const int SC_MINIMIZE = 0xF020;
     if (m.Msg == WM_SYSCOMMAND && ((int)m.WParam == SC_MINIMIZE || (int)m.WParam == SC_CLOSE))
     {
         //最小化到系统栏 
         this.Hide();
         return;
     }
     base.WndProc(ref    m);
}

----------------------------------------------
不规则窗体拖动

using System.Runtime.InteropServices;

[DllImport("user32.dll")]
public static extern bool ReleaseCapture();
[DllImport("user32.dll")]
public static extern bool SendMessage(IntPtr hwnd, int wMsg, int wParam, int lParam);
public const int WM_SYSCOMMAND = 0x0112;
public const int SC_MOVE = 0xF010;
public const int HTCAPTION = 0x0002;

public void ShapedForm_MouseDown(object sender, MouseEventArgs e) 

ReleaseCapture();
SendMessage(this.Handle, WM_SYSCOMMAND, SC_MOVE + HTCAPTION, 0);
}

-------------------------------------------------------------------
protected override void WndProc(ref Message m)
  {
  //拦截窗体最小化按钮消息,调用隐藏动画并隐藏窗体
  if (m.Msg == (int)hyFrameWork.win32Api.Enum.WinMsg.WM_SYSCOMMAND)
  {
  if (m.WParam.ToInt32() == (int)hyFrameWork.win32Api.Enum.ECNCSysCommandConstants.SC_MINIMIZE)
  {
  formShowControl(false);
  return;
  }
  }
  base.WndProc(ref m);
  }
  -------------------------------------------
  大体的消息说明,你自己看吧 
WM_NULL = $0000; 
WM_CREATE = $0001; 
应用程序创建一个窗口 
WM_DESTROY = $0002; 
一个窗口被销毁 
WM_MOVE = $0003; 
移动一个窗口 
WM_SIZE = $0005; 
改变一个窗口的大小 
WM_ACTIVATE = $0006; 
一个窗口被激活或失去激活状态; 
WM_SETFOCUS = $0007; 
获得焦点后 
WM_KILLFOCUS = $0008; 
失去焦点 
WM_ENABLE = $000A; 
改变enable状态 
WM_SETREDRAW = $000B; 
设置窗口是否能重画 
WM_SETTEXT = $000C; 
应用程序发送此消息来设置一个窗口的文本 
WM_GETTEXT = $000D; 
应用程序发送此消息来复制对应窗口的文本到缓冲区 
WM_GETTEXTLENGTH = $000E; 
得到与一个窗口有关的文本的长度(不包含空字符) 
WM_PAINT = $000F; 
要求一个窗口重画自己 
WM_CLOSE = $0010; 
当一个窗口或应用程序要关闭时发送一个信号 
WM_QUERYENDSESSION = $0011; 
当用户选择结束对话框或程序自己调用ExitWindows函数 
WM_QUIT = $0012; 
用来结束程序运行或当程序调用postquitmessage函数 
WM_QUERYOPEN = $0013; 
当用户窗口恢复以前的大小位置时,把此消息发送给某个图标 
WM_ERASEBKGND = $0014; 
当窗口背景必须被擦除时(例在窗口改变大小时) 
WM_SYSCOLORCHANGE = $0015; 
当系统颜色改变时,发送此消息给所有顶级窗口 
WM_ENDSESSION = $0016; 
当系统进程发出WM_QUERYENDSESSION消息后,此消息发送给应用程序, 
通知它对话是否结束 
WM_SYSTEMERROR = $0017; 
WM_SHOWWINDOW = $0018; 
当隐藏或显示窗口是发送此消息给这个窗口 
WM_ACTIVATEAPP = $001C; 
发此消息给应用程序哪个窗口是激活的,哪个是非激活的; 
WM_FONTCHANGE = $001D; 
当系统的字体资源库变化时发送此消息给所有顶级窗口 
WM_TIMECHANGE = $001E; 
当系统的时间变化时发送此消息给所有顶级窗口 
WM_CANCELMODE = $001F; 
发送此消息来取消某种正在进行的摸态(操作) 
WM_SETCURSOR = $0020; 
如果鼠标引起光标在某个窗口中移动且鼠标输入没有被捕获时,就发消息给某个窗口 
WM_MOUSEACTIVATE = $0021; 
当光标在某个非激活的窗口中而用户正按着鼠标的某个键发送此消息给当前窗口 
WM_CHILDACTIVATE = $0022; 
发送此消息给MDI子窗口当用户点击此窗口的标题栏,或当窗口被激活,移动,改变大小 
WM_QUEUESYNC = $0023; 
此消息由基于计算机的训练程序发送,通过WH_JOURNALPALYBACK的hook程序 
分离出用户输入消息 
WM_GETMINMAXINFO = $0024; 
此消息发送给窗口当它将要改变大小或位置; 
WM_PAINTICON = $0026; 
发送给最小化窗口当它图标将要被重画 
WM_ICONERASEBKGND = $0027; 
此消息发送给某个最小化窗口,仅当它在画图标前它的背景必须被重画 
WM_NEXTDLGCTL = $0028; 
发送此消息给一个对话框程序去更改焦点位置 
WM_SPOOLERSTATUS = $002A; 
每当打印管理列队增加或减少一条作业时发出此消息 
WM_DRAWITEM = $002B; 
当button,combobox,listbox,menu的可视外观改变时发送 
此消息给这些空件的所有者 
WM_MEASUREITEM = $002C; 
当button, combo box, list box, list view control, or menu item 被创建时 
发送此消息给控件的所有者 
WM_DELETEITEM = $002D; 
当the list box 或 combo box 被销毁 或 当 某些项被删除通过LB_DELETESTRING, LB_RESETCONTENT, CB_DELETESTRING, or CB_RESETCONTENT 消息 
WM_VKEYTOITEM = $002E; 
此消息有一个LBS_WANTKEYBOARDINPUT风格的发出给它的所有者来响应WM_KEYDOWN消息 
WM_CHARTOITEM = $002F; 
此消息由一个LBS_WANTKEYBOARDINPUT风格的列表框发送给他的所有者来响应WM_CHAR消息 
WM_SETFONT = $0030; 
当绘制文本时程序发送此消息得到控件要用的颜色 
WM_GETFONT = $0031; 
应用程序发送此消息得到当前控件绘制文本的字体 
WM_SETHOTKEY = $0032; 
应用程序发送此消息让一个窗口与一个热键相关连 
WM_GETHOTKEY = $0033; 
应用程序发送此消息来判断热键与某个窗口是否有关联 
WM_QUERYDRAGICON = $0037; 
此消息发送给最小化窗口,当此窗口将要被拖放而它的类中没有定义图标,应用程序能 
返回一个图标或光标的句柄,当用户拖放图标时系统显示这个图标或光标 
WM_COMPAREITEM = $0039; 
发送此消息来判定combobox或listbox新增加的项的相对位置 
WM_GETOBJECT = $003D; 
WM_COMPACTING = $0041; 
显示内存已经很少了 
WM_WINDOWPOSCHANGING = $0046; 
发送此消息给那个窗口的大小和位置将要被改变时,来调用setwindowpos函数或其它窗口管理函数 
WM_WINDOWPOSCHANGED = $0047; 
发送此消息给那个窗口的大小和位置已经被改变时,来调用setwindowpos函数或其它窗口管理函数 
WM_POWER = $0048;(适用于16位的windows) 
当系统将要进入暂停状态时发送此消息 
WM_COPYDATA = $004A; 
当一个应用程序传递数据给另一个应用程序时发送此消息 
WM_CANCELJOURNAL = $004B; 
当某个用户取消程序日志激活状态,提交此消息给程序 
WM_NOTIFY = $004E; 
当某个控件的某个事件已经发生或这个控件需要得到一些信息时,发送此消息给它的父窗口 
WM_INPUTLANGCHANGEREQUEST = $0050; 
当用户选择某种输入语言,或输入语言的热键改变 
WM_INPUTLANGCHANGE = $0051; 
当平台现场已经被改变后发送此消息给受影响的最顶级窗口 
WM_TCARD = $0052; 
当程序已经初始化windows帮助例程时发送此消息给应用程序 
WM_HELP = $0053; 
此消息显示用户按下了F1,如果某个菜单是激活的,就发送此消息个此窗口关联的菜单,否则就 
发送给有焦点的窗口,如果当前都没有焦点,就把此消息发送给当前激活的窗口 
WM_USERCHANGED = $0054; 
当用户已经登入或退出后发送此消息给所有的窗口,当用户登入或退出时系统更新用户的具体 
设置信息,在用户更新设置时系统马上发送此消息; 
WM_NOTIFYFORMAT = $0055; 
公用控件,自定义控件和他们的父窗口通过此消息来判断控件是使用ANSI还是UNICODE结构 
在WM_NOTIFY消息,使用此控件能使某个控件与它的父控件之间进行相互通信 
WM_CONTEXTMENU = $007B; 
当用户某个窗口中点击了一下右键就发送此消息给这个窗口 
WM_STYLECHANGING = $007C; 
当调用SETWINDOWLONG函数将要改变一个或多个 窗口的风格时发送此消息给那个窗口 
WM_STYLECHANGED = $007D; 
当调用SETWINDOWLONG函数一个或多个 窗口的风格后发送此消息给那个窗口 
WM_DISPLAYCHANGE = $007E; 
当显示器的分辨率改变后发送此消息给所有的窗口 
WM_GETICON = $007F; 
此消息发送给某个窗口来返回与某个窗口有关连的大图标或小图标的句柄; 
WM_SETICON = $0080; 
程序发送此消息让一个新的大图标或小图标与某个窗口关联; 
WM_NCCREATE = $0081; 
当某个窗口第一次被创建时,此消息在WM_CREATE消息发送前发送; 
WM_NCDESTROY = $0082; 
此消息通知某个窗口,非客户区正在销毁 
WM_NCCALCSIZE = $0083; 
当某个窗口的客户区域必须被核算时发送此消息 
WM_NCHITTEST = $0084;//移动鼠标,按住或释放鼠标时发生 
WM_NCPAINT = $0085; 
程序发送此消息给某个窗口当它(窗口)的框架必须被绘制时; 
WM_NCACTIVATE = $0086; 
此消息发送给某个窗口 仅当它的非客户区需要被改变来显示是激活还是非激活状态; 
WM_GETDLGCODE = $0087; 
发送此消息给某个与对话框程序关联的控件,widdows控制方位键和TAB键使输入进入此控件 
通过响应WM_GETDLGCODE消息,应用程序可以把他当成一个特殊的输入控件并能处理它 
WM_NCMOUSEMOVE = $00A0; 
当光标在一个窗口的非客户区内移动时发送此消息给这个窗口 //非客户区为:窗体的标题栏及窗 
的边框体 
WM_NCLBUTTONDOWN = $00A1; 
当光标在一个窗口的非客户区同时按下鼠标左键时提交此消息 
WM_NCLBUTTONUP = $00A2; 
当用户释放鼠标左键同时光标某个窗口在非客户区十发送此消息; 
WM_NCLBUTTONDBLCLK = $00A3; 
当用户双击鼠标左键同时光标某个窗口在非客户区十发送此消息 
WM_NCRBUTTONDOWN = $00A4; 
当用户按下鼠标右键同时光标又在窗口的非客户区时发送此消息 
WM_NCRBUTTONUP = $00A5; 
当用户释放鼠标右键同时光标又在窗口的非客户区时发送此消息 
WM_NCRBUTTONDBLCLK = $00A6; 
当用户双击鼠标右键同时光标某个窗口在非客户区十发送此消息 
WM_NCMBUTTONDOWN = $00A7; 
当用户按下鼠标中键同时光标又在窗口的非客户区时发送此消息 
WM_NCMBUTTONUP = $00A8; 
当用户释放鼠标中键同时光标又在窗口的非客户区时发送此消息 
WM_NCMBUTTONDBLCLK = $00A9; 
当用户双击鼠标中键同时光标又在窗口的非客户区时发送此消息 
WM_KEYFIRST = $0100; 
WM_KEYDOWN = $0100; 
//按下一个键 
WM_KEYUP = $0101; 
//释放一个键 
WM_CHAR = $0102; 
//按下某键,并已发出WM_KEYDOWN, WM_KEYUP消息 
WM_DEADCHAR = $0103; 
当用translatemessage函数翻译WM_KEYUP消息时发送此消息给拥有焦点的窗口 
WM_SYSKEYDOWN = $0104; 
当用户按住ALT键同时按下其它键时提交此消息给拥有焦点的窗口; 
WM_SYSKEYUP = $0105; 
当用户释放一个键同时ALT 键还按着时提交此消息给拥有焦点的窗口 
WM_SYSCHAR = $0106; 
当WM_SYSKEYDOWN消息被TRANSLATEMESSAGE函数翻译后提交此消息给拥有焦点的窗口 
WM_SYSDEADCHAR = $0107; 
当WM_SYSKEYDOWN消息被TRANSLATEMESSAGE函数翻译后发送此消息给拥有焦点的窗口 
WM_KEYLAST = $0108; 
WM_INITDIALOG = $0110; 
在一个对话框程序被显示前发送此消息给它,通常用此消息初始化控件和执行其它任务 
WM_COMMAND = $0111; 
当用户选择一条菜单命令项或当某个控件发送一条消息给它的父窗口,一个快捷键被翻译 
WM_SYSCOMMAND = $0112; 
当用户选择窗口菜单的一条命令或当用户选择最大化或最小化时那个窗口会收到此消息 
WM_TIMER = $0113; //发生了定时器事件 
WM_HSCROLL = $0114; 
当一个窗口标准水平滚动条产生一个滚动事件时发送此消息给那个窗口,也发送给拥有它的控件 
WM_VSCROLL = $0115; 
当一个窗口标准垂直滚动条产生一个滚动事件时发送此消息给那个窗口也,发送给拥有它的控件 WM_INITMENU = $0116; 
当一个菜单将要被激活时发送此消息,它发生在用户菜单条中的某项或按下某个菜单键,它允许 
程序在显示前更改菜单 
WM_INITMENUPOPUP = $0117; 
当一个下拉菜单或子菜单将要被激活时发送此消息,它允许程序在它显示前更改菜单,而不要 
改变全部 
WM_MENUSELECT = $011F; 
当用户选择一条菜单项时发送此消息给菜单的所有者(一般是窗口) 
WM_MENUCHAR = $0120; 
当菜单已被激活用户按下了某个键(不同于加速键),发送此消息给菜单的所有者; 
WM_ENTERIDLE = $0121; 
当一个模态对话框或菜单进入空载状态时发送此消息给它的所有者,一个模态对话框或菜单进入空载状态就是在处理完一条或几条先前的消息后没有消息它的列队中等待 
WM_MENURBUTTONUP = $0122; 
WM_MENUDRAG = $0123; 
WM_MENUGETOBJECT = $0124; 
WM_UNINITMENUPOPUP = $0125; 
WM_MENUCOMMAND = $0126; 
WM_CHANGEUISTATE = $0127; 
WM_UPDATEUISTATE = $0128; 
WM_QUERYUISTATE = $0129; 
WM_CTLCOLORMSGBOX = $0132; 
在windows绘制消息框前发送此消息给消息框的所有者窗口,通过响应这条消息,所有者窗口可以 
通过使用给定的相关显示设备的句柄来设置消息框的文本和背景颜色 
WM_CTLCOLOREDIT = $0133; 
当一个编辑型控件将要被绘制时发送此消息给它的父窗口;通过响应这条消息,所有者窗口可以 
通过使用给定的相关显示设备的句柄来设置编辑框的文本和背景颜色 
WM_CTLCOLORLISTBOX = $0134; 
当一个列表框控件将要被绘制前发送此消息给它的父窗口;通过响应这条消息,所有者窗口可以 
通过使用给定的相关显示设备的句柄来设置列表框的文本和背景颜色 
WM_CTLCOLORBTN = $0135; 
当一个按钮控件将要被绘制时发送此消息给它的父窗口;通过响应这条消息,所有者窗口可以 
通过使用给定的相关显示设备的句柄来设置按纽的文本和背景颜色 
WM_CTLCOLORDLG = $0136; 
当一个对话框控件将要被绘制前发送此消息给它的父窗口;通过响应这条消息,所有者窗口可以 
通过使用给定的相关显示设备的句柄来设置对话框的文本背景颜色 
WM_CTLCOLORSCROLLBAR= $0137; 
当一个滚动条控件将要被绘制时发送此消息给它的父窗口;通过响应这条消息,所有者窗口可以 
通过使用给定的相关显示设备的句柄来设置滚动条的背景颜色 
WM_CTLCOLORSTATIC = $0138; 
当一个静态控件将要被绘制时发送此消息给它的父窗口;通过响应这条消息,所有者窗口可以 
通过使用给定的相关显示设备的句柄来设置静态控件的文本和背景颜色 
WM_MOUSEFIRST = $0200; 
WM_MOUSEMOVE = $0200; 
// 移动鼠标 
WM_LBUTTONDOWN = $0201; 
//按下鼠标左键 
WM_LBUTTONUP = $0202; 
//释放鼠标左键 
WM_LBUTTONDBLCLK = $0203; 
//双击鼠标左键 
WM_RBUTTONDOWN = $0204; 
//按下鼠标右键 
WM_RBUTTONUP = $0205; 
//释放鼠标右键 
WM_RBUTTONDBLCLK = $0206; 
//双击鼠标右键 
WM_MBUTTONDOWN = $0207; 
//按下鼠标中键 
WM_MBUTTONUP = $0208; 
//释放鼠标中键 
WM_MBUTTONDBLCLK = $0209; 
//双击鼠标中键 
WM_MOUSEWHEEL = $020A; 
当鼠标轮子转动时发送此消息个当前有焦点的控件 
WM_MOUSELAST = $020A; 
WM_PARENTNOTIFY = $0210; 
当MDI子窗口被创建或被销毁,或用户按了一下鼠标键而光标在子窗口上时发送此消息给它的父窗口 
WM_ENTERMENULOOP = $0211; 
发送此消息通知应用程序的主窗口that已经进入了菜单循环模式 
WM_EXITMENULOOP = $0212; 
发送此消息通知应用程序的主窗口that已退出了菜单循环模式 
WM_NEXTMENU = $0213; 
WM_SIZING = 532; 
当用户正在调整窗口大小时发送此消息给窗口;通过此消息应用程序可以监视窗口大小和位置 
也可以修改他们 
WM_CAPTURECHANGED = 533; 
发送此消息 给窗口当它失去捕获的鼠标时; 
WM_MOVING = 534; 
当用户在移动窗口时发送此消息,通过此消息应用程序可以监视窗口大小和位置 
也可以修改他们; 
WM_POWERBROADCAST = 536; 
此消息发送给应用程序来通知它有关电源管理事件; 
WM_DEVICECHANGE = 537; 
当设备的硬件配置改变时发送此消息给应用程序或设备驱动程序 
WM_IME_STARTCOMPOSITION = $010D; 
WM_IME_ENDCOMPOSITION = $010E; 
WM_IME_COMPOSITION = $010F; 
WM_IME_KEYLAST = $010F; 
WM_IME_SETCONTEXT = $0281; 
WM_IME_NOTIFY = $0282; 
WM_IME_CONTROL = $0283; 
WM_IME_COMPOSITIONFULL = $0284; 
WM_IME_SELECT = $0285; 
WM_IME_CHAR = $0286; 
WM_IME_REQUEST = $0288; 
WM_IME_KEYDOWN = $0290; 
WM_IME_KEYUP = $0291; 
WM_MDICREATE = $0220; 
应用程序发送此消息给多文档的客户窗口来创建一个MDI 子窗口 
WM_MDIDESTROY = $0221; 
应用程序发送此消息给多文档的客户窗口来关闭一个MDI 子窗口 
WM_MDIACTIVATE = $0222; 
应用程序发送此消息给多文档的客户窗口通知客户窗口激活另一个MDI子窗口,当客户窗口收到 
此消息后,它发出WM_MDIACTIVE消息给MDI子窗口(未激活)激活它; 
WM_MDIRESTORE = $0223; 
程序 发送此消息给MDI客户窗口让子窗口从最大最小化恢复到原来大小 
WM_MDINEXT = $0224; 
程序 发送此消息给MDI客户窗口激活下一个或前一个窗口 
WM_MDIMAXIMIZE = $0225; 
程序发送此消息给MDI客户窗口来最大化一个MDI子窗口; 
WM_MDITILE = $0226; 
程序 发送此消息给MDI客户窗口以平铺方式重新排列所有MDI子窗口 
WM_MDICASCADE = $0227; 
程序 发送此消息给MDI客户窗口以层叠方式重新排列所有MDI子窗口 
WM_MDIICONARRANGE = $0228; 
程序 发送此消息给MDI客户窗口重新排列所有最小化的MDI子窗口 
WM_MDIGETACTIVE = $0229; 
程序 发送此消息给MDI客户窗口来找到激活的子窗口的句柄 
WM_MDISETMENU = $0230; 
程序 发送此消息给MDI客户窗口用MDI菜单代替子窗口的菜单 
WM_ENTERSIZEMOVE = $0231; 
WM_EXITSIZEMOVE = $0232; 
WM_DROPFILES = $0233; 
WM_MDIREFRESHMENU = $0234; 
WM_MOUSEHOVER = $02A1; 
WM_MOUSELEAVE = $02A3; 
WM_CUT = $0300; 
程序发送此消息给一个编辑框或combobox来删除当前选择的文本 
WM_COPY = $0301; 
程序发送此消息给一个编辑框或combobox来复制当前选择的文本到剪贴板 
WM_PASTE = $0302; 
程序发送此消息给editcontrol或combobox从剪贴板中得到数据 
WM_CLEAR = $0303; 
程序发送此消息给editcontrol或combobox清除当前选择的内容; 
WM_UNDO = $0304; 
程序发送此消息给editcontrol或combobox撤消最后一次操作 
WM_RENDERFORMAT = $0305;

WM_RENDERALLFORMATS = $0306; 
WM_DESTROYCLIPBOARD = $0307; 
当调用ENPTYCLIPBOARD函数时 发送此消息给剪贴板的所有者 
WM_DRAWCLIPBOARD = $0308; 
当剪贴板的内容变化时发送此消息给剪贴板观察链的第一个窗口;它允许用剪贴板观察窗口来 
显示剪贴板的新内容; 
WM_PAINTCLIPBOARD = $0309; 
当剪贴板包含CF_OWNERDIPLAY格式的数据并且剪贴板观察窗口的客户区需要重画; 
WM_VSCROLLCLIPBOARD = $030A; 
WM_SIZECLIPBOARD = $030B; 
当剪贴板包含CF_OWNERDIPLAY格式的数据并且剪贴板观察窗口的客户区域的大小已经改变是此消息通过剪贴板观察窗口发送给剪贴板的所有者; 
WM_ASKCBFORMATNAME = $030C; 
通过剪贴板观察窗口发送此消息给剪贴板的所有者来请求一个CF_OWNERDISPLAY格式的剪贴板的名字 
WM_CHANGECBCHAIN = $030D; 
当一个窗口从剪贴板观察链中移去时发送此消息给剪贴板观察链的第一个窗口; 
WM_HSCROLLCLIPBOARD = $030E; 
此消息通过一个剪贴板观察窗口发送给剪贴板的所有者 ;它发生在当剪贴板包含CFOWNERDISPALY格式的数据并且有个事件在剪贴板观察窗的水平滚动条上;所有者应滚动剪贴板图象并更新滚动条的值; 
WM_QUERYNEWPALETTE = $030F; 
此消息发送给将要收到焦点的窗口,此消息能使窗口在收到焦点时同时有机会实现他的逻辑调色板 
WM_PALETTEISCHANGING= $0310; 
当一个应用程序正要实现它的逻辑调色板时发此消息通知所有的应用程序 
WM_PALETTECHANGED = $0311; 
此消息在一个拥有焦点的窗口实现它的逻辑调色板后发送此消息给所有顶级并重叠的窗口,以此 
来改变系统调色板 
WM_HOTKEY = $0312; 
当用户按下由REGISTERHOTKEY函数注册的热键时提交此消息 
WM_PRINT = 791; 
应用程序发送此消息仅当WINDOWS或其它应用程序发出一个请求要求绘制一个应用程序的一部分; 
WM_PRINTCLIENT = 792; 
WM_HANDHELDFIRST = 856; 
WM_HANDHELDLAST = 863; 
WM_PENWINFIRST = $0380; 
WM_PENWINLAST = $038F; 
WM_COALESCE_FIRST = $0390; 
WM_COALESCE_LAST = $039F; 
WM_DDE_FIRST = $03E0; 
WM_DDE_INITIATE = WM_DDE_FIRST + 0; 
一个DDE客户程序提交此消息开始一个与服务器程序的会话来响应那个指定的程序和主题名; 
WM_DDE_TERMINATE = WM_DDE_FIRST + 1; 
一个DDE应用程序(无论是客户还是服务器)提交此消息来终止一个会话; 
WM_DDE_ADVISE = WM_DDE_FIRST + 2; 
一个DDE客户程序提交此消息给一个DDE服务程序来请求服务器每当数据项改变时更新它 
WM_DDE_UNADVISE = WM_DDE_FIRST + 3; 
一个DDE客户程序通过此消息通知一个DDE服务程序不更新指定的项或一个特殊的剪贴板格式的项 
WM_DDE_ACK = WM_DDE_FIRST + 4; 
此消息通知一个DDE(动态数据交换)程序已收到并正在处理WM_DDE_POKE, WM_DDE_EXECUTE, WM_DDE_DATA, WM_DDE_ADVISE, WM_DDE_UNADVISE, or WM_DDE_INITIAT消息 
WM_DDE_DATA = WM_DDE_FIRST + 5; 
一个DDE服务程序提交此消息给DDE客户程序来传递个一数据项给客户或通知客户的一条可用数据项 
WM_DDE_REQUEST = WM_DDE_FIRST + 6; 
一个DDE客户程序提交此消息给一个DDE服务程序来请求一个数据项的值; 
WM_DDE_POKE = WM_DDE_FIRST + 7; 
一个DDE客户程序提交此消息给一个DDE服务程序,客户使用此消息来请求服务器接收一个未经同意的数据项;服务器通过答复WM_DDE_ACK消息提示是否它接收这个数据项; 
WM_DDE_EXECUTE = WM_DDE_FIRST + 8; 
一个DDE客户程序提交此消息给一个DDE服务程序来发送一个字符串给服务器让它象串行命令一样被处理,服务器通过提交WM_DDE_ACK消息来作回应; 
WM_DDE_LAST = WM_DDE_FIRST + 8; 
WM_APP = $8000; 
WM_USER = $0400; 
此消息能帮助应用程序自定义私有消息; 
///////////////////////////////////////////////////////////////////// 
通知消息(Notification message)是指这样一种消息,一个窗口内的子控件发生了一些事情,需要通 
知父窗口。通知消息只适用于标准的窗口控件如按钮、列表框、组合框、编辑框,以及Windows 95公 
共控件如树状视图、列表视图等。例如,单击或双击一个控件、在控件中选择部分文本、操作控件的 
滚动条都会产生通知消息。 
按扭 
B N _ C L I C K E D //用户单击了按钮 
B N _ D I S A B L E //按钮被禁止 
B N _ D O U B L E C L I C K E D //用户双击了按钮 
B N _ H I L I T E //用户加亮了按钮 
B N _ PA I N T按钮应当重画 
B N _ U N H I L I T E加亮应当去掉 
组合框 
C B N _ C L O S E U P组合框的列表框被关闭 
C B N _ D B L C L K用户双击了一个字符串 
C B N _ D R O P D O W N组合框的列表框被拉出 
C B N _ E D I T C H A N G E用户修改了编辑框中的文本 
C B N _ E D I T U P D AT E编辑框内的文本即将更新 
C B N _ E R R S PA C E组合框内存不足 
C B N _ K I L L F O C U S组合框失去输入焦点 
C B N _ S E L C H A N G E在组合框中选择了一项 
C B N _ S E L E N D C A N C E L用户的选择应当被取消 
C B N _ S E L E N D O K用户的选择是合法的 
C B N _ S E T F O C U S组合框获得输入焦点 
编辑框 
E N _ C H A N G E编辑框中的文本己更新 
E N _ E R R S PA C E编辑框内存不足 
E N _ H S C R O L L用户点击了水平滚动条 
E N _ K I L L F O C U S编辑框正在失去输入焦点 
E N _ M A X T E X T插入的内容被截断 
E N _ S E T F O C U S编辑框获得输入焦点 
E N _ U P D AT E编辑框中的文本将要更新 
E N _ V S C R O L L用户点击了垂直滚动条消息含义 
列表框 
L B N _ D B L C L K用户双击了一项 
L B N _ E R R S PA C E列表框内存不够 
L B N _ K I L L F O C U S列表框正在失去输入焦点 
L B N _ S E L C A N C E L选择被取消 
L B N _ S E L C H A N G E选择了另一项 
L B N _ S E T F O C U S列表框获得输入焦点 
------------------------------------------------------------------
static bool flag = false;
        protected override void WndProc(ref Message m)//C# 重写 WndProc 移动窗口
        {
           switch (m.Msg) {
                 case 0x0200: //WM_MOUSEMOVE
                         if (flag)
                         {
                             PostMessage(this.Handle, 0x00A1,new IntPtr(2),m.LParam);//这里需要导入 user32.dll 
                         }
                         break;
                     case 0x201://WM_LBUTTONDOWN
                         flag = true;
                         break;
                     case 0x202://WM_LBUTTONUP
                        flag = false;

break;

default:
                    base.WndProc(ref m);
                    break;
            }

[DllImport("user32.dll", CharSet = CharSet.Unicode)]
        public static extern IntPtr PostMessage(IntPtr hwnd, int wMsg, IntPtr wParam, IntPtr lParam);
------------------------------------------------------------
 [DllImport("user32.dll")]
        public static extern IntPtr SendMessage(IntPtr hWnd, int msg, int wparam, int lparam);
        protected override void OnMouseDown(MouseEventArgs e)
        {
            base.OnMouseDown(e); if (e.Button == MouseButtons.Left)//按下的是鼠标左键  
            {
                Capture = false;//释放鼠标,使能够手动操作         
                SendMessage(Handle, 0x00A1, 2, 0);//拖动窗体     
            }
        }
 --------------------------------------------------------
  protected override void WndProc(ref Message m)
        {
            switch (m.Msg)
            {
                case 0x0201://鼠标左键按下的消息
                    m.Msg = 0x00A1;//更改消息为非客户区按下鼠标
                    m.LParam = IntPtr.Zero;//默认值
                    m.WParam = new IntPtr(2);//鼠标放在标题栏内
                    break;
            }
            base.WndProc(ref m);
        }
 ----------------------------------------------------

重写 WndProc函数来同时实现无标题栏的窗体移动和禁止双击窗体最大化

protected override void WndProc(ref Message m)
        {
            const int WM_NCHITTEST = 0x84;
            const int HTCLIENT = 0x01;
            const int HTCAPTION = 0x02;
            const int WM_SYSCOMMAND = 0x112;
            const int SC_MAXMIZE = 0xF030;
            const int WM_NCLBUTTONDBLCLK = 0xA3;
            switch (m.Msg)
            {
                case 0x4e:
                case 0xd:
                case 0xe:
                case 0x14:
                    base.WndProc(ref m);
                    break;
                case WM_NCHITTEST://鼠标点任意位置后可以拖动窗体
                    
                    this.DefWndProc(ref m);
                    if (m.Result.ToInt32() == HTCLIENT)
                    {
                        m.Result = new IntPtr(HTCAPTION);
                        return;
                    }
                    break;
                case WM_NCLBUTTONDBLCLK://禁止双击最大化
                    Console.WriteLine(this.WindowState);
                    
                        return;

break;

default:
                    
                    base.WndProc(ref m);
                    break;
            }
        }
----------------------------------------------------------
第一步,先在类的级别中申明两个API函数,

  ///注册热键 
  [DllImport("user32.dll")] 
  private static extern bool RegisterHotKey(IntPtr hWnd, int id, uint fsModifiers, Keys vk); 
  ///卸载热键 
  [DllImport("user32.dll")] 
  private static extern bool UnregisterHotKey(IntPtr hWnd, int id);

  注意引入名字空间 System.Runtime.InteropServices;

  上面两个函数参数的解释:

  RegisterHotKey 函数中,

  hWnd为本程序窗口的句柄,在C#的窗口中直接用Handle属性就可以引用窗口句柄

  id为热键的标示符,是我们自己定义的,因为一个程序中可以定义多个热键,所以要用这个字段来区别,具体用法见下

  fsModifiers为激活热键时,是否和系统键组合使用,none:0 Alt:1 Ctrl:2 Shift:4,并且还可以用或运算来组合使用

  vk就是要定义的热键,C#中按键都被包含在Keys枚举中

  UnregisterHotKey 函数中,

  hWnd和RegisterHotKey 函数中是一样的,id为要卸载的热键标示

  第二步,在窗口的初始化中注册热键,例如 RegisterHotKey(Handle, 100, 0, Keys.F9);

  将些热键识别为100,0表示不使用系统键,Keys.F9表示此热键为F9,若要同时按Shift+Ctrl+F9,则第三个参数应该为 2|4 其实对二进制运算熟悉的朋友,               可以立刻算出 2|4=6

  第三步,可以响应热键了:

  重写窗体的WndProc函数,具体代码为

  protected override void WndProc(ref Message m) { 
  //这个if的条件中固定的 
  if (m.Msg == 0x312) { 
  //这个if就在于判断是哪个热键,100对应上面RegisterHotKey函数中的第二个参数 
  if (m.WParam.ToInt32() == 100) { 
  //这里就是响应的函数 
  DoSomething(); 
  } 
  } 
  base.WndProc(ref m); 
  }

  WndProc()函数的功能就是处理Windows消息,在其它地方还将看到这个函数的妙用!

  第四步,在程序结束的时候调用UnregisterHotKey(Handle,100)卸载此热键!

  WIN32

  第一步,在int WINAPI WinMain()函数中,

  while(GetMessage(&msg,NULL,0,0)){……}

  之前,注册热键

  RegisterHotKey(hWnd,100,0,VK_F9)

  VK_F9为WinUser.h中定义的宏,其他键也是用类似的形式进行定义,不过数字键和字母键,直接用'0','A'这样的形式

  第二步,在while(GetMessage(&msg,NULL,0,0)) {……}

  循环体内,添加代码

  //这是响应热键条件,从WM_HOTKEY宏的定义中,可以看出来它代表的值为 0x312 ,和C#中的if条件是一致的 
  if(msg.message ==WM_HOTKEY) 
  { 
  //这里的条件用来判断热键的标示符 
  if(msg.wParam==100) 
  { 
  //这里调用想要执行的函数 
  DoSomething(); 
  } 
  }

  第三步,在while()循环结束后,调用 UnregisterHotKey(hWnd,100) 卸载热键。

  MFC

  第一步,在MFC的窗体中添加其WM_HOTKEY消息的响应,

  void C热键Dlg::OnHotKey(UINT nHotKeyId, UINT nKey1, UINT nKey2) 
  { 
  //这里用来判断热键的标识符 
  if(nHotKeyId == 100) 
  { 
  //响应函数 
  DoSomething(); 
  } 
  CDialog::OnHotKey(nHotKeyId, nKey1, nKey2); 
  }

  第二步,在MFC窗体的初始化地方,添加注册热键的代码:

  RegisterHotKey(m_hWnd,100,0,VK_F9);

  注意上面的字段m_hWnd,是CWnd类中字段,完成由MFC控制,我们只需要在适合地方引用就可以了。

  第三步,在窗体销毁的地方调用 UnregisterHotKey(hWnd,100) 卸载热键就可以了。

  到这里,大家可以已经掌握了在三种环境中设置系统级热键的方法。笔者的体会是,在WIN32和C#中方法大概相同,笔者就是在C#的基础上,尝试着在WIN32中实现该功能,结果一次成功,但是在MFC中就走了一些弯路。
------------------------------------------------------------------------

protected override void WndProc(ref Message m)
        {
            switch (m.Msg)
            {
                case 0x0005://change size: WM_SIZE
                    {
                        FormWindowState newState = FormWindowState.Normal;
                        switch (m.WParam.ToInt32())
                        {
                            case 0://SIZE_RESTORED
                                newState = FormWindowState.Normal;
                                break;
                            case 1://SIZE_MINIMIZED
                                newState = FormWindowState.Minimized;
                                break;
                            case 2://SIZE_MAXIMIZED
                                newState = FormWindowState.Maximized;
                                break;
                            default:
                                break;
                        }

if (newState != this.preWindowState)
                        {
                            this.OnWindowStateChanged(new WindowStateChangedEventArgs(this.preWindowState, newState));
                            this.preWindowState = newState;
                        }
                    }
                    break;
                default:
                    break;
            }
            base.WndProc(ref m);
        }
        #endregion
    }
}

c#委托与事件、消息、WndProc用法(转)的更多相关文章

  1. 关于C# 委托(delegate)与事件(event)的用法及事例

    C#中的委托和事件对于新手可能会有一点难理解,所以先从一个小例子入手,以便能更好的理解其如何使用.有一个学生每天定闹钟在早上6点起床,所以当每天早上6点的时候,闹钟就会响起来,从而学生才会按时起床. ...

  2. .NET面试题系列[7] - 委托与事件

    委托和事件 委托在C#中具有无比重要的地位. C#中的委托可以说俯拾即是,从LINQ中的lambda表达式到(包括但不限于)winform,wpf中的各种事件都有着委托的身影.C#中如果没有了事件,那 ...

  3. .NET基础拾遗(4)委托、事件、反射与特性

    Index : (1)类型语法.内存管理和垃圾回收基础 (2)面向对象的实现和异常的处理基础 (3)字符串.集合与流 (4)委托.事件.反射与特性 (5)多线程开发基础 (6)ADO.NET与数据库开 ...

  4. 【详细】【转】C#中理解委托和事件 事件的本质其实就是委托 RabbitMQ英汉互翼(一),RabbitMQ, RabbitMQ教程, RabbitMQ入门

    [详细][转]C#中理解委托和事件   文章是很基础,但很实用,看了这篇文章,让我一下回到了2016年刚刚学委托的时候,故转之! 1.委托 委托类似于C++中的函数指针(一个指向内存位置的指针).委托 ...

  5. [转载]C#委托和事件(Delegate、Event、EventHandler、EventArgs)

    原文链接:http://blog.csdn.net/zwj7612356/article/details/8272520 14.1.委托 当要把方法作为实参传送给其他方法的形参时,形参需要使用委托.委 ...

  6. C#委托与事件

    一.在控制台下使用委托和事件 我们都知道,C#中有"接口"这个概念,所谓的"接口"就是定义一套标准,然后由实现类来具体实现其中的方法,所以说"接口,是 ...

  7. C#之委托与事件

    委托与事件 废话一堆:网上关于委托.事件的文章有很多,一千个哈姆雷特就有一千个莎士比亚,以下内容均是本人个人见解. 1. 委托 1.1 委托的使用 这一小章来学习一下怎么简单的使用委托,了解一些基本的 ...

  8. 重读C#委托、事件有感

    去年学习C#的时候按照进度把C#的委托和事件“认真”的学习了一下,自己知道委托和事件的重要性,所以也努力的学习,可说实也没怎么学懂.碰巧这段时间在做一个解析GPS数据的小项目,因为其中有需要条件自动判 ...

  9. C#编程之委托与事件四(二)【转】

    C#编程之委托与事件(二)       我在上一篇文章(C#编程之委托与事件(一) )中通过示例结合的方法介绍了委托,在本文中,我同样以代码示例的方式来介绍C#里的事件机制. 二.事件   1.了解概 ...

随机推荐

  1. curl获取图片

    <?php set_time_limit(0); //执行30秒超时后继续执行 header("Content-type:text/html;charset=utf-8"); ...

  2. 伪元素after,before,css/js控制样式

    CSS<style> body { font: 200%/1.45 charter; } ref::before { content: '\00A7'; letter-spacing: . ...

  3. BZOJ2038:[2009国家集训队]小Z的袜子

    浅谈莫队:https://www.cnblogs.com/AKMer/p/10374756.html 题目传送门:https://lydsy.com/JudgeOnline/problem.php?i ...

  4. BZOJ3296:[USACO2011OPEN]Learning Language

    浅谈并查集:https://www.cnblogs.com/AKMer/p/10360090.html 题目传送门:https://lydsy.com/JudgeOnline/problem.php? ...

  5. CF884D:Boxes And Balls

    浅谈\(Huffman\)树:https://www.cnblogs.com/AKMer/p/10300870.html 题目传送门:https://codeforces.com/problemset ...

  6. 蓝桥杯 算法训练 ALGO-124 数字三角形

    算法训练 数字三角形   时间限制:1.0s   内存限制:256.0MB   问题描述 (图3.1-1)示出了一个数字三角形. 请编一个程序计算从顶至底的某处的一条路 径,使该路径所经过的数字的总和 ...

  7. java代码。从来没想过java里的继承是多么的难懂。如哲学

    总结:实例.. 这里不加super("aaa",32); 运行:父类和子类的姓名,年龄是一样的.那这个super为什么没效果呢? 显示:class:Ji姓名是 小红年龄是:20课程 ...

  8. GUI练习中

    总结:JFrame和Frame是有很大差别的. 不要混淆.否则方法是不能成功调用的 特别是背景色:JFrame.对象f在main里无法调用背景色前景色都不想显示 一下是书上的一段代码,编译错误,但是可 ...

  9. [0day]jQuery Mobile XSS

    漏洞影响范围: 任何一个website使用了 jQuery Mobile 并且开放了重定向都有可能存在XSS,并且目前还没有相关补丁信息. 应用介绍: jQuery Mobile是jQuery 框架的 ...

  10. 2015.3.4 VS2005调用MFC dll时报错及解决

    今天在用VS2005调用MFCdll时报错: 正试图在 os 加载程序锁内执行托管代码.不要尝试在 DllMain 或映像初始化函数内运行托管代码... 原因是我在dll的CSpaceApp::CSp ...