.NET图像界面按钮的clicked事件浅谈

但是事件并不局限于图像界面,也可能是满足程序的某个逻辑判断触发了事件。引发事件的对象叫做事件发送方,捕获事件并对其做出响应的对象叫做事件接收方。但是事件发送方并不知道哪个对象或者方法会处理它引发的事件,所以需要在事件发送方和事件接收方之间存在一媒介,明确某个对象或者某个类型的某个方法会对这个事件进行响应。.NET用委托作为事件发送方与事件接收方之间的媒介,委托只有一个签名,只要方法签名与委托签名匹配的方法,都可以声明自己对这个委托类型的事件的感兴趣,接收并处理它。
事件发送方的对象为了给事件接收对象传递一些额外信息,就需要写一个派生于System.EventArgs的类,封装一些数据。


代码如下:

public class DrawEventArgs:EventArgs
    {
        privatedouble m_Size;
        public DrawEventArgs(doublesize)
        {
            m_Size = size;
        }
        public double Size
        {
            get
            {
                returnm_Size;
            }
        }
    }

下面声明一委托,该委托有两个参数,一个是参数代表事件的发送方,另一个是参数是该事件封装的数据。
public  delegate void ShapeSizeChanged(objectsender,DrawEventArgs e);
ShapeSizeChanged的实例可以绑定到任何和它方法签名匹配的方法上。
下面是自定义事件委托,
public event ShapeSizeChanged SizeChanged;
下面是一个负责引发事件的类型代码,


代码如下:

public class DrawManager
    {
        public event ShapeSizeChangedSizeChanged;
        protectedvoid OnSizeChanged(DrawEventArgse)
        {
            ShapeSizeChangedtemp = SizeChanged;
            //是否有委托与该事件关联
            if(temp != null)
            {
                temp(this,e);
            }
        }
        public void SizeChange(doublesize)
        {
            DrawEventArgse = new DrawEventArgs(size);
            OnSizeChanged(e);
        }
    }

而后定义两个监听事件的类型,


代码如下:

public class Square
    {
        publicSquare(DrawManager drawManager)
        {
            //drawManager.SizeChanged+= DrawSquare;
            //把事件关联到委托上
            drawManager.SizeChanged += new ShapeSizeChanged(DrawSquare);
        }
        public void DrawSquare(objectsender, DrawEventArgs e)
        {
            Console.WriteLine(string.Format("TheSquare'Length = {0}", e.Size));
        }
        public void Detach(DrawManagerdrawManager)
        {
            //drawManager.SizeChanged-= DrawSquare;
            //解除事件和委托的关联
            drawManager.SizeChanged -= new ShapeSizeChanged(DrawSquare);
        }
    }
    public class Rectangle
    {
        publicRectangle(DrawManager drawManager)
        {
            drawManager.SizeChanged +=DrawRectangle;
        }
        public void DrawRectangle(objectsender, DrawEventArgs e)
        {
            Console.WriteLine(string.Format("TheRectangle'length={0} and width={1}.",e.Size*2,e.Size));
        }
        public void Detach(DrawManagerdrawManager)
        {
            drawManager.SizeChanged -=DrawRectangle;
        }
    }

测试代码,


代码如下:

class Program
    {
        static void Main(string[]args)
        {
            DrawManagerdrawManager = new DrawManager();
            Rectanglerect = new Rectangle(drawManager);
            Squaresquare = new Square(drawManager);
            //引发事件
            drawManager.SizeChange(5);

//解除监听事件
            square.Detach(drawManager);
            drawManager.SizeChange(10);
            Console.ReadLine();
        }
    }
/*运行结果
  The Rectangle'length=10 and width=5.
  The Square'Length = 5
  The Rectangle'length=20 and width=10.
 */

.NET中的事件模式和观察者模式非常相似,也可谓是观察者模式在.NET下的进化版吧,下面用观察者模式实现上面功能以做对比,首先定义两个接口,IObserver和IObservable,如下


代码如下:

public interface IObserver
    {
        voidNotify(DrawEventArgs e);
    }
    public interface IObservable
    {
        voidRegister(IObserver observer);
        voidUnRegister(IObserver observer);
    }

下面是改写后的两个观察者类,


代码如下:

public class NewRectangle:IObserver
    {
        privateObserverManager m;
        publicNewRectangle(ObserverManager oManager)
        {
            m=oManager;
            oManager.Register(this);
        }
        public void Notify(DrawEventArgse)
        {
            Console.WriteLine(string.Format("TheRectangle'length={0} and width={1}.", e.Size * 2, e.Size));
        }
        public void Detach()
        {
            m.UnRegister(this);
        }
    }
    public class NewSquare:IObserver
    {
        privateObserverManager m;

publicNewSquare(ObserverManager oManager)
        {
            m=oManager;
            oManager.Register(this);
        }
        public void Notify(DrawEventArgse)
        {
            Console.WriteLine(string.Format("TheSquare'Length = {0}.", e.Size));
        }
        public void Detach()
        {
            m.UnRegister(this);
        }
    }

下面是负责通知观察者的类型,


代码如下:

public class ObserverManager:IObservable
    {
        protectedArrayList arrList;
        publicObserverManager()
        {
            arrList = newArrayList();
        }
        public void Register(IObserverobserver)
        {
            arrList.Add(observer);
        }
        public void UnRegister(IObserverobserver)
        {
            if(arrList.Contains(observer))
            {
                arrList.Remove(observer);
            }
        }
        public void NotifyObservers(doublesize)
        {
            DrawEventArgse = new DrawEventArgs(size);
            foreach(IObserver observer inarrList)
            {
                observer.Notify(e);
            }
        }
        public void SizeChanged(doublesize)
        {
            NotifyObservers(size);
        }
    }

下面是调用代码,


代码如下:

static void Main(string[]args)
        {
            ObserverManageroManager = new ObserverManager();
            NewRectanglerect = new NewRectangle(oManager);
            NewSquaresquare = new NewSquare(oManager);
            oManager.SizeChanged(5);
            square.Detach();
            oManager.SizeChanged(10);
     Console.ReadLine();
        }

最好运行下代码,这样可以更easy的理解这两种模式微妙的差别了。
对事件来说,还可以显式的用add和remove编写事件访问器,事件访问器通常有编译器生成,所以可以显式的用事件访问器修改DrawManager类型,


代码如下:

public class DrawManager
    {
        privateevent ShapeSizeChangedm_SizeChanged;
        privatereadonly objectm_lock = new object();
        public event ShapeSizeChangedSizeChanged
        {
            add
            {
                lock(m_lock)
                {
                    m_SizeChanged += value;
                }
            }
            remove
            {
                lock(m_lock)
                {
                    m_SizeChanged -= value;
                }
            }
        }
        protectedvoid OnSizeChanged(DrawEventArgse)
        {
            ShapeSizeChangedtemp = m_SizeChanged;
            //是否有委托与该事件关联
            if(temp != null)
            {
                temp(this,e);
            }
        }
        public void SizeChange(doublesize)
        {
            DrawEventArgse = new DrawEventArgs(size);
            OnSizeChanged(e);
        }
    }

(0)

相关推荐

  • .NET事件监听机制的局限与扩展分析

    本文实例分析了.NET事件监听机制的局限与扩展.分享给大家供大家参考.具体分析如下: .NET中把"事件"看作一个基本的编程概念,并提供了非常优美的语法支持,对比如下C#和Java代码可以看出两种语言设计思想之间的差异. 复制代码 代码如下: // C# someButton.Click += OnSomeButtonClick; 复制代码 代码如下: // Java someButton.addActionListener(     new ActionListener(){    

  • asp.net页面触发事件panel滚动条高度不变的实现方法

    此文是为解决asp.net页面按钮点击触发事件后panel滚动条非自动回到顶端的解决方案,对于页面触发一个事件后,panel滚动条重新回到顶端,做下面的工作每次都要往下拉一下,特别是选择TreeView的时候,这个问题非常头疼,受到this.MaintainScrollPositionOnPostBack = true;的启示有感而发. 原理是在点击时保存panel滚动条的位置,页面触发事件完成后加载时重新赋值.保证页面每次点击都保存,就包含所有的按钮和别的点击事件. 步骤: 1.脚本 <hea

  • asp.net在事件中启动线程来打开一个页面的实现方法

    在页面点击一个按钮,其目的是在按钮中做两件事情,一件需要点击按钮马上完成,另一件事情是点击按钮后做其他事情.如果按顺序一次做完感觉特别耗时,下面简单罗列一下. protected void Button1_Click(object sender, EventArgs e) { Label1.Text = TextBox1.Text; //在这做第一件事情 dowork(); //做完后马上启动线程 System.Threading.Thread thread = new System.Threa

  • ASP.NET中实现定制自己的委托和事件参数类

    本文实例讲述了ASP.NET中实现定制自己的委托和事件参数类的方法,对于学习ASP.NET有很好的参考借鉴价值.具体方法如下: 一般在实际开发中,对于事件不需要传递数据信息时,像上面的KingTextBox控件的事件,在引发事件时传递的参数为EventArgs.Empty,如下所示: OnTextChanged(EventArgs.Empty); 这是因为控件KingTextBox的TextChanged事件比较简单,这里不需要参数对象传递数据.但像一些复杂的控件比如GridView的按钮命令事

  • ASP.NET 页面事件执行顺序介绍

    复制代码 代码如下: #region 控件事件 第六步 protected void Button1_Click(object sender, EventArgs e) { //用这些事件来处理特定控件事件,如 Button 控件的 Click 事件或 TextBox 控件的 TextChanged 事件. //注意 //在回发请求中,如果页包含验证程序控件,请在执行任何处理之前检查 Page 和各个验证控件的 IsValid 属性. } #endregion #region OnLoadCom

  • .net自定义事件示例分享

    1.新建一个控制台应用程序TestDelegate,本项目主要实现:热水器加热,报警器监控,当热水温度达到80度的时候报警器报警这样一个简单的事件处理程序 2.定义委托处理程序 复制代码 代码如下: public delegate void PlayGameHandler(object sender, System.EventArgs e); 3.添加一个报警器类,报警方法只有在温度超过80度的时候会被调用 复制代码 代码如下: // 负责报警的人    public class 报警器   

  • .NET中基于事件的异步模式-EAP

    前言 在C# 5.0中,新增了async await 2个关键字支持异步编程的操作.在讲述这两个关键字之前,我先总结一下.NET中的常见的异步编程模型. 异步编程一直是比较复杂的问题,其中要处理多线程之间的数据同步.获取进度.可取消.获取结果.不影响主线程操作.多个任务之间互相不影响等,因此需要设计编程模型去处理此类问题. 从.NET 4.5开始,支持的三种异步编程模式: 基于事件的异步编程设计模式 (EAP,Event-based Asynchronous Pattern) 异步编程模型(AP

  • .NET WinFrom中给文本框添加拖放事件的代码

    在开发.NET WinForm程序时,有时候我们希望窗体上的文本框能接受鼠标拖放事件,比如允许将文件直接拖放到文本框中以直接获取到文件的本地路径,或者将选取的字符串直接拖放到文本框中等等.要实现这个功能其实很简单,代码如下. 1. 将文本框的属性AllowDrop设置成True 2. 给文本框添加DragEnter事件 复制代码 代码如下: private void textBox1_DragEnter(object sender, DragEventArgs e) { if (e.Data.G

  • .net,js捕捉文本框回车键事件的小例子(兼容多浏览器)

    js: 复制代码 代码如下: function bubufx_enterpress(e) {         var keynum; if (window.event) // IE           {             keynum = e.keyCode; }         else if (e.which) // Netscape/Firefox/Opera           {             keynum = e.which;         }         i

  • 详解ASP.NET页面生命周期事件

    下面是ASP.NET页面初始的过程:1. Page_Init();2. Load ViewState;3. Load Postback data;4. Page_Load();5. Handle control events;6. Page_PreRender();7. Page_Render();8. Unload event;9. Dispose method called; 下面对其中的一些过程作下描述:1. Page_Init();这个过程主要是初始化控件,每次页面载入执行这个初始过程,

  • .Net WInform开发笔记(五)关于事件Event

    我前面几篇博客中提到过.net中的事件与Windows事件的区别,本文讨论的是前者,也就是我们代码中经常用到的Event.Event很常见,Button控件的Click.KeyPress等等,PictureBox控件的Paint等等都属于本文讨论范畴,本文会例举出有关"事件编程"的几种方法,还会提及由"事件编程"引起的MemoryLeak(跟"内存泄露"差不多),以及由"事件编程"引起的一些异常. 引子: .net中事件最常用

随机推荐