Asp.net控件开发学习笔记(九)----服务器控件事件

简介:
EventCollection

如果在单个控件中有多个事件,那么使用System.ComponentModel.EventHandlerList对事件进行保存将会在内存占用上有不错的提高。EventHandlerList对一个类内发布多个事件提供了一个列表容器。下面是多个事件和使用EventHandlerList的对比示意:



 第一步是实例化一个EventHandlerList的实例:

protected EventHandlerList eventList = new EventHandlerList();

第二步是声明一个容器用于保存事件的key

private static readonly object ClickEvent = new object();

最后一步是像往常一样声明一个事件,但有所不同的是就像属性的get和set程序块一样,对于事件C#提供了add和remove关键字:

public event EventHandler Click

        {

            add

            {

                Events.AddHandler(ClickEvent, value);

            }

            remove

            {

                Events.RemoveHandler(ClickEvent, value);

            }

      }

 而在这时的事件调用方法就会像下面代码:

protected virtual void OnClick(EventArgs e)

        {

            EventHandler clickEventDelegate = (EventHandler)Events[ClickEvent];

            if (clickEventDelegate != null)

             {

                clickEventDelegate(this, e);

            }

     }

 上面代码首先从事件列表中通过索引器以第一步中保存事件的key为参数提取出事件并检查客户端是否注册到此事件,如果是,则激发事件。

 

Command事件和事件冒泡

    Command事件是System.Web.UI.WebControls命名空间里的强大模式。这个最好的例子是GridView


    

 在GridView的Row里嵌套的button点击会触发Command事件,后台可以根据CommandArgument的不同来决定是执行edit操作还是delete操作等。而事件冒泡有些像javascript里的事件冒泡,但有所不同的是这里的事件冒泡到能够处理这个事件的地方停止,比如上图中command事件会冒泡到DataGrid里的ItemCommand里停止,因为ItemCommand事件可以对command事件进行处理.

 在定义Command事件时会和前面大同小异,不同之处在于首先需要一个继承与System.EventArgs的CommandEventArgs类来进行参数传递,代码如下

public class CommandEventArgs : EventArgs

{

    public CommandEventArgs(string _commandName,string _commandArgument)

{

    CommandName=_commandName;

    CommandArgument=_commandArgument;

}

       private string commandname;

        private string commandArgument;

        public virtual string CommandName

        {

            get

            {

                return commandname;

            }

            set

            {

                commandname = value;

            }

        }

        public virtual string CommandArgument

        {

            get

            {

                return commandArgument;

            }

            set

            {

                commandArgument = value;

            }

        }   

}

然后在需要定义的控件里定义这两个属性,代码如下:

         public virtual string CommandName

         {

            get

            {

                object name = ViewState["CommandName"];

                if (name == null)

                    return string.Empty;

                else

                    return (string)name;

            }

            set

            {

                ViewState["CommandName"] = value;

            }

        }

        public virtual string CommandArgument

        {

            get

            {

                object arg = ViewState["CommandArgument"];

                if (arg == null)

                    return string.Empty;

                else

                    return (string)arg;

            }

            set

            {

                ViewState["CommandArgument"] = value;

            }

        }

然后重复前面的步骤,在控件内部定义命令事件:

private static readonly object CommandKey = new object();

        public event CommandEventHandler Command

        {

            add

            {

                Events.AddHandler(CommandKey, value);

            }

            remove

            {

                Events.RemoveHandler(CommandKey, value);

            }

        }

最后一步和前面说的引发事件的OnXXX的实现都略有不同,这里在控件内部实现的代码如下:

       protected virtual void OnCommand(CommandEventArgs ce)

        {

            CommandEventHandler commandEventDelegate =(CommandEventHandler)Events[CommandKey];

            if (commandEventDelegate != null)

            {

                commandEventDelegate(this, ce);

            }

            RaiseBubbleEvent(this, ce);

        }

 注意最后一句,RaiseBubbleEvent方法.这个方法可以将控件的事件传递到它的父容器上。

 到这里很多人都会好奇,那CommandName和CommandArgument两个参数是如何传入到CommandEventArgs里去的呢?

 其实是在引发事件时传入的,代码如下:

  OnCommand(new CommandEventArgs(CommandName, CommandArgument));

 

DEMO 带Command事件的Button

    其实这个Demo就是把上面的代码全部拼装起来,代码可能会有点长,代码如下:

    namespace DemoButton

{

    [ToolboxData("<{0}:superbutton runat=server></{0}:superbutton>")]

    public class ButtonDemo : Control, IPostBackEventHandler

    {

        public delegate void CommandEventHandler(object sender,CommandEventArgs e);

        public virtual string Text

        {

            get

            {

                object text = ViewState["Text"];

                if (text == null)

                    return string.Empty;

                else

                    return (string)text;

            }

            set

            {

                ViewState["Text"] = value;

            }

        }

        private static readonly object ClickKey = new object();

        public event EventHandler Click

        {

            add

            {

                Events.AddHandler(ClickKey, value);

            }

            remove

            {

                Events.RemoveHandler(ClickKey, value);

            }

        }

        protected virtual void OnClick(EventArgs e)

        {

            EventHandler clickEventDelegate = (EventHandler)Events[ClickKey];

            if (clickEventDelegate != null)

            {

                clickEventDelegate(this, e);

            }

        }

        private static readonly object CommandKey = new object();

        public event CommandEventHandler Command

        {

            add

            {

                Events.AddHandler(CommandKey, value);

            }

            remove

            {

                Events.RemoveHandler(CommandKey, value);

            }

        }

        public virtual string CommandName

        {

            get

            {

                object name = ViewState["CommandName"];

                if (name == null)

                    return string.Empty;

                else

                    return (string)name;

            }

            set

            {

                ViewState["CommandName"] = value;

            }

        }

        public virtual string CommandArgument

        {

            get

            {

                object arg = ViewState["CommandArgument"];

                if (arg == null)

                    return string.Empty;

                else

                    return (string)arg;

            }

            set

            {

                ViewState["CommandArgument"] = value;

            }

        }

        protected virtual void OnCommand(CommandEventArgs ce)

        {

            CommandEventHandler commandEventDelegate = (CommandEventHandler)Events[CommandKey];

            if (commandEventDelegate != null)

            {

                commandEventDelegate(this, ce);

            }

            RaiseBubbleEvent(this, ce);

        }

        public void RaisePostBackEvent(string argument)

        {

            OnCommand(new CommandEventArgs(CommandName, CommandArgument));

            //OnClick(EventArgs.Empty);

        }

        protected override void Render(HtmlTextWriter writer)

        {

            base.Render(writer);

            Page.VerifyRenderingInServerForm(this);

            writer.Write("<INPUT type=""submit""");

            writer.Write(" name=""" + this.UniqueID + """");

            writer.Write(" id=""" + this.UniqueID + """");

            writer.Write(" value=""" + Text + """");

            writer.Write(" />");

        }

    }

 

    public class CommandEventArgs : EventArgs

    {

        public CommandEventArgs(string _commandName, string _commandArgument)

        {

            CommandName = _commandName;

            CommandArgument = _commandArgument;

        }

        private string commandname;

        private string commandArgument;

        public virtual string CommandName

        {

            get

            {

                return commandname;

            }

            set

            {

                commandname = value;

            }

        }

        public virtual string CommandArgument

        {

            get

            {

                return commandArgument;

            }

            set

            {

                commandArgument = value;

            }

        }

    }

}

前台代码:

首先注册页面控件:

 <%@ Register Namespace="DemoButton" TagPrefix="cc" %>

前台代码:

   <cc:ButtonDemo runat="server" Text="第一个按钮" ID="bt1" CommandName="bt1"

        CommandArgument="第一个button的参数" oncommand="bt1_Command" ></cc:ButtonDemo>

      

 <cc:ButtonDemo runat="server" Text="第二个按钮" ID="bt2" CommandName="bt1"

        CommandArgument="第二个button的参数" oncommand="bt1_Command" ></cc:ButtonDemo>

事件处理程序:

    protected void bt1_Command(object sender, DemoButton.CommandEventArgs e)

    {

        if (e.CommandName == "bt1")

        {

            Response.Write("第一个button被点击了,参数是"+e.CommandArgument);

        }

        else if (e.CommandArgument == "bt12")

        {

            Response.Write("第二个button被点击了,参数是" + e.CommandArgument);

        }

}

 

Demo的结果很简单,就不演示了:-)


本文转自CareySon博客园博客,原文链接:http://www.cnblogs.com/CareySon/archive/2009/10/14/1582906.html,如需转载请自行联系原作者
相关文章
|
2月前
|
设计模式 开发框架 JavaScript
基于.NET8 + Vue/UniApp前后端分离的快速开发框架,开箱即用!
基于.NET8 + Vue/UniApp前后端分离的快速开发框架,开箱即用!
|
3月前
|
XML JSON API
ServiceStack:不仅仅是一个高性能Web API和微服务框架,更是一站式解决方案——深入解析其多协议支持及简便开发流程,带您体验前所未有的.NET开发效率革命
【10月更文挑战第9天】ServiceStack 是一个高性能的 Web API 和微服务框架,支持 JSON、XML、CSV 等多种数据格式。它简化了 .NET 应用的开发流程,提供了直观的 RESTful 服务构建方式。ServiceStack 支持高并发请求和复杂业务逻辑,安装简单,通过 NuGet 包管理器即可快速集成。示例代码展示了如何创建一个返回当前日期的简单服务,包括定义请求和响应 DTO、实现服务逻辑、配置路由和宿主。ServiceStack 还支持 WebSocket、SignalR 等实时通信协议,具备自动验证、自动过滤器等丰富功能,适合快速搭建高性能、可扩展的服务端应用。
171 3
|
6天前
|
缓存 算法 安全
精选10款C#/.NET开发必备类库(含使用教程),工作效率提升利器!
精选10款C#/.NET开发必备类库(含使用教程),工作效率提升利器!
38 12
|
4天前
|
Linux API C#
基于 .NET 开发的多功能流媒体管理控制平台
基于 .NET 开发的多功能流媒体管理控制平台
|
4天前
|
Web App开发 前端开发 调度
一款基于 .NET + Blazor 开发的智能访客管理系统
一款基于 .NET + Blazor 开发的智能访客管理系统
|
4天前
|
前端开发 JavaScript C#
基于.NET8+Vue3开发的权限管理&个人博客系统
基于.NET8+Vue3开发的权限管理&个人博客系统
|
4天前
|
网络协议 C#
基于.NET WinForm开发的一款硬件及协议通讯工具
基于.NET WinForm开发的一款硬件及协议通讯工具
|
4天前
|
监控 前端开发 API
一款基于 .NET MVC 框架开发、功能全面的MES系统
一款基于 .NET MVC 框架开发、功能全面的MES系统
|
2月前
|
开发框架 .NET C#
在 ASP.NET Core 中创建 gRPC 客户端和服务器
本文介绍了如何使用 gRPC 框架搭建一个简单的“Hello World”示例。首先创建了一个名为 GrpcDemo 的解决方案,其中包含一个 gRPC 服务端项目 GrpcServer 和一个客户端项目 GrpcClient。服务端通过定义 `greeter.proto` 文件中的服务和消息类型,实现了一个简单的问候服务 `GreeterService`。客户端则通过 gRPC 客户端库连接到服务端并调用其 `SayHello` 方法,展示了 gRPC 在 C# 中的基本使用方法。
45 5
在 ASP.NET Core 中创建 gRPC 客户端和服务器
|
2月前
|
传感器 人工智能 供应链
.NET开发技术在数字化时代的创新作用,从高效的开发环境、强大的性能表现、丰富的库和框架资源等方面揭示了其关键优势。
本文深入探讨了.NET开发技术在数字化时代的创新作用,从高效的开发环境、强大的性能表现、丰富的库和框架资源等方面揭示了其关键优势。通过企业级应用、Web应用及移动应用的创新案例,展示了.NET在各领域的广泛应用和巨大潜力。展望未来,.NET将与新兴技术深度融合,拓展跨平台开发,推动云原生应用发展,持续创新。
46 4