当前位置:  编程技术>.net/c#/asp.net

C# 中的动态创建组件(属性及事件)的实现思路及方法

    来源: 互联网  发布时间:2014-10-25

    本文导语:  通常在写程序的时候,当要用到某些组件,采用的方法一般都是动态创建,用完以后就释放掉。Visual   C#在程序运行的时候也可以动态创建组件,下面就结合一个程序例子来具体介绍如何用Visual   C#动态生成组件。首先让我...

通常在写程序的时候,当要用到某些组件,采用的方法一般都是动态创建,用完以后就释放掉。Visual   C#在程序运行的时候也可以动态创建组件,下面就结合一个程序例子来具体介绍如何用Visual   C#动态生成组件。首先让我们了解一下,在动态创建组件的过程中要用到的一些概论和理论。
一.   Boxing   (装箱)和Unboxing   (出箱):
在用Visual   C#动态创建组件的时候,要涉及到二种数据类型变量的转换,这二种类型变量就是实值类型(Value   Type)变量和参考类型(Reference   Type)变量,而这种转换过程在Visual   C#中被称为Boxing   (装箱)和Unboxing   (出箱)。其中把实值类型变量转换成参考类型变量就是Boxing   (装箱);把参考类型变量转换成实值类型变量就是Unboxing   (出箱)。那么什么是实值类型,说的简单些,就是我们平常使用的整型、布尔型、枚举型等,这些类型的变量就是实值类型变量了;所谓参考类型,在Visual   C#中指的就是Object、Class、Interface、Delegate、String、Array等,他和实值类型最主要的不同之处就是,参考类型变量存放的是指向实体对象的指针,而实值类型变量却是实实在在地实体对象。在本文介绍的程序中,主要涉及的是出箱。具体的处理方法,在下面有着具体介绍。

二.   程序设计中的关键步骤以及解决方法:
文中软件主要功能是用通过窗体上的二个按钮来创建二个不同类型的WinForm组件--Button组件和TextBox组件,并在创建的同时为每一个组件的属性赋值,给每一个创建的组件也创建了事件。
1).如何在窗体上创建Button组件:
其实用Visual   C#创建一个组件是十分方便的,只用下列二行语句就可以完成了:

代码如下:

//创建一个新的Button组件
Button   myButton   =   new   Button   (   )   ;
//在窗体中显示此按钮
this.Controls.Add   (   myButton   )   ;

但此时创建的这个Button组件没有任何属性,并且也没有任何事件,在本文中介绍的程序中创建的Button组件,不仅有属性也有事件,下列语句就是本文程序创建Button组件源代码:
代码如下:

//按钮数量计算器在每次按钮按动后加 "1 "
counter   +=   1   ;
//对要产生的按钮的纵坐标的相对位置是前一个产生按钮的相对位置的纵坐标加 "3 "
locY   +=   this.btnAdd.Height   +   3   ;
//创建一个新的Button组件
Button   myButton   =   new   Button   (   )   ;
//设定他的名称和Text属性,以及产生的相对位置
myButton.Name   =   "Button   "   +   counter   ;
myButton.Text   =   "按钮   "   +   counter   ;
myButton.Location   =   new   Point   (   btnAdd.Location.X   ,   locY   )   ;  
//为产生的新的Button组件设定事件,本文中为产生的按钮设定了三个事件
myButton.MouseEnter   +=   new   System.EventHandler   (   this.btn_MouseEnter   )   ;
myButton.MouseLeave   +=   new   System.EventHandler   (   this.btn_MouseLeave   )   ;
myButton.Click   +=   new   System.EventHandler   (   this.btn_Click   )   ;
//在窗体中显示此按钮
this.Controls.Add   (   myButton   )   ;  

程序不仅为每一个组件的属性都赋值,而且为每一个组件都创建了三个事件。细心的读者可能已经注意到,程序为每一个组件创建的事件的名称都是一样的。这样就有一个问题,如何在这一样的事件中,识别到底是哪个Button组件触发了事件。
(2).确定是哪个组件触发了事件:
由于程序中为每一个创建的Button组件的事件都是一样的,要想正确处理这些组件的事件,就需要在事件触发的程序中判断到底是哪个组件触发了这个事件。这就需要用到上面所提出的装箱和出箱。我们知道Sender对象是一个参考类型变量,他存放的是指向触发当前事件实体对象的指针。要把他给转换成实值对象类型,通过下列语句就可以确定是哪个组件触发了当前事件:
代码如下:

private   void   btn_MouseEnter   (   object   sender   ,   System.EventArgs   e   )
{
//出箱
Button   currentButton   =   (   Button   )   sender   ;
//设定按钮的背景色
currentButton.BackColor   =   Color.Red   ;
}  

其他事件可以仿照此事件的处理过程来处理。
(3).   如何在窗体上创建TextBox组件:
创建TextBox组件的过程和创建Button组件过程相类似,只是在创建的组件类型上面有一点区别,具体实现语句如下:
代码如下:

//文本框数量计算器在每次按钮按动后加 "1 "
counter01   +=   1   ;
//对要产生的文本框的纵坐标的相对位置是前一个产生按钮的相对位置的纵坐标加 "3
locY1   +=   this.txtAdd.Height   +   3   ;
//创建一个新的TextBox组件
TextBox   myBox   =   new   TextBox   (   )   ;
//设定他的名称和Text属性,以及产生的位置
myBox.Name   =   "TextBox   "   +   counter01   ;
myBox.Text   =   "文本框   "   +   counter01   ;
myBox.Location   =   new   Point   (   txtAdd.Location.X   ,   locY1   )   ;  
//为产生的新的TextBox组件设定事件,本文中为产生的文本框设定了一个事件
myBox.Click   +=   new   System.EventHandler   (   this.btn_Click   )   ;
//在窗体中显示此文本框
this.Controls.Add   (   myBox   )   ;  

此时细心的读者又会发现,为每一个TextBox组件创建Click事件和为Button组件创建的Click事件也是一样的,这样在Click事件中不仅要判断是哪个组件触发了事件,还要判断是那种类型的组件触发了事件,下面语句是实现这些判断地具体方法:
代码如下:

private   void   btn_Click   (   object   sender   ,   System.EventArgs   e   )
{
if   (   sender.GetType   (   )   ==   typeof   (   Button   )   )  
{
Button   control   =   (   Button   )   sender   ;
MessageBox.Show   (   control.Text   +   "被按动了! ");
}
else  
{
TextBox   control   =   (   TextBox   )   sender   ;  
MessageBox.Show   (   control.Text   +   "被按动了! "   )   ;
}
}  

当然如果你也可以单独为TextBox组件创建Click事件。此时创建的事件语句可改为:
代码如下:

myBox.Click   +=   new   System.EventHandler   (   this.txt   _Click   )   ;  

//下面是实现txt   _Click   (   )事件的程序代码:
private   void   txt_Click   (   object   sender   ,   System.EventArgs   e   )
{
TextBox   currentButton   =   (   TextBox   )   sender   ;
MessageBox.Show   (   currentButton.Text   +   "被按动了! ");
}

下面是实现上面结果的程序源代码:

代码如下:

using   System   ;
using   System.Drawing   ;
using   System.Collections   ;
using   System.ComponentModel   ;
using   System.Windows.Forms   ;
using   System.Data   ;
namespace   DynamicControls
{
public   class   Form1   :   Form
{
private   Button   btnAdd   ;
private   System.ComponentModel.Container   components   =   null   ;
private   Button   txtAdd   ;
//给产生的按钮定义一个数量计算器
private   int   counter   ;
//给产生的按钮定义相对位置的纵坐标
private   int   locY   ;
//给产生的文本框定义一个数量计算器
private   int   counter01   ;
//给产生的文本框定义相对位置的纵坐标
private   int   locY1   ;
public   Form1   (   )
{
InitializeComponent   (   )   ;
//初始化产生的按钮何文本框位置的纵坐标
locY   =   this.btnAdd.Location.Y   ;
locY1   =   this.txtAdd.Location.Y   ;
}

//清除在程序中使用到的资源
protected   override   void   Dispose   (   bool   disposing   )
{
if   (   disposing   )
{
if   (   components   !=   null   )  
{
components.Dispose   (   )   ;
}
}
base.Dispose   (   disposing   )   ;
}

private   void   InitializeComponent   (   )
{
this.btnAdd   =   new   Button   (   )   ;
this.txtAdd   =   new   Button   (   )   ;
this.SuspendLayout   (   )   ;

this.btnAdd.FlatStyle   =   FlatStyle.Popup   ;
this.btnAdd.Location   =   new   System.Drawing.Point   (   8   ,   16   )   ;
this.btnAdd.Name   =   "btnAdd "   ;
this.btnAdd.TabIndex   =   0   ;
this.btnAdd.Text   =   "生成按钮! "   ;
this.btnAdd.Click   +=   new   System.EventHandler   (   this.btnAdd_Click   )   ;

this.txtAdd.FlatStyle   =   FlatStyle.Popup   ;
this.txtAdd.Location   =   new   System.Drawing.Point   (   108   ,   16   )   ;
this.txtAdd.Name   =   "txtAdd "   ;
this.txtAdd.TabIndex   =   1   ;
this.txtAdd.Text   =   "生成文本框! "   ;
this.txtAdd.Click   +=   new   System.EventHandler   (   this.txtAdd_Click   )   ;

this.AutoScaleBaseSize   =   new   System.Drawing.Size   (   5   ,   13   )   ;
this.ClientSize   =   new   System.Drawing.Size   (   292   ,   273   )   ;
this.Controls.Add   (   btnAdd   )   ;
this.Controls.Add   (   txtAdd   )   ;
this.Name   =   "Form1 "   ;
this.Text   =   "在Visual   C#中如何动态产生组件! "   ;
this.ResumeLayout   (   false   )   ;  

}
static   void   Main   (   )  
{
Application.Run   (   new   Form1   (   )   )   ;
}
private   void   btnAdd_Click   (   object   sender   ,   System.EventArgs   e   )
{
//按钮数量计算器在每次按钮按动后加 "1 "
counter   +=   1   ;
//对要产生的按钮的纵坐标的相对位置是前一个产生按钮的相对位置的纵坐标加 "3 "
locY   +=   this.btnAdd.Height   +   3   ;
//创建一个新的Button组件
Button   myButton   =   new   Button   (   )   ;
//设定他的名称和Text属性,以及产生的位置
myButton.Name   =   "Button   "   +   counter   ;
myButton.Text   =   "按钮   "   +   counter   ;
myButton.Location   =   new   Point   (   btnAdd.Location.X   ,   locY   )   ;  

//为产生的新的Button组件设定事件,本文中为产生的按钮设定了三个事件
myButton.MouseEnter   +=   new   System.EventHandler   (   this.btn_MouseEnter   )   ;
myButton.MouseLeave   +=   new   System.EventHandler   (   this.btn_MouseLeave   )   ;
myButton.Click   +=   new   System.EventHandler   (   this.btn_Click   )   ;
//在窗体中显示此按钮
this.Controls.Add   (   myButton   )   ;
}

private   void   txtAdd_Click   (   object   sender   ,   System.EventArgs   e   )
{
//文本框数量计算器在每次按钮按动后加 "1 "
counter01   +=   1   ;
//对要产生的文本框的纵坐标的相对位置是前一个产生按钮的相对位置的纵坐标加 "3
locY1   +=   this.txtAdd.Height   +   3   ;
//创建一个新的TextBox组件
TextBox   myBox   =   new   TextBox   (   )   ;
//设定他的名称和Text属性,以及产生的位置
myBox.Name   =   "TextBox   "   +   counter01   ;
myBox.Text   =   "文本框   "   +   counter01   ;
myBox.Location   =   new   Point   (   txtAdd.Location.X   ,   locY1   )   ;  
//为产生的新的TextBox组件设定事件,本文中为产生的文本框设定了一个事件
myBox.Click   +=   new   System.EventHandler   (   this.btn_Click   )   ;
//在窗体中显示此文本框
this.Controls.Add   (   myBox   )   ;
}
private   void   btn_MouseEnter   (   object   sender   ,   System.EventArgs   e   )
{
//出箱
Button   currentButton   =   (   Button   )   sender   ;
//设定按钮的背景色
currentButton.BackColor   =   Color.Red   ;
}

private   void   btn_MouseLeave   (   object   sender   ,   System.EventArgs   e   )
{
//出箱
Button   currentButton   =   (   Button   )   sender   ;
currentButton.BackColor   =   Control.DefaultBackColor   ;
}

private   void   btn_Click   (   object   sender   ,   System.EventArgs   e   )
{
if   (   sender.GetType   (   )   ==   typeof   (   Button   )   )  
{
Button   control   =   (   Button   )   sender   ;
MessageBox.Show   (   control.Text   +   "被按动了! ");
}
else  
{
TextBox   control   =   (   TextBox   )   sender   ;  
MessageBox.Show   (   control.Text   +   "被按动了! "   )   ;
}
}

}
}  


四.   总结:
通过上面介绍,不难看出,动态创建组件并不是一件很难的事情,难就难在为这个组件创建事件上面,因为这涉及到实值类型变量和参考类型变量的转换,这就是所谓的装箱和出箱的问题。当然在程序设计的时候,你不仅可以创建那些可见的组件,也可以创建那些不可见的组件,具体的实现方法和本文中的方法类似。

    
 
 

您可能感兴趣的文章:

  • C#实现动态显示及动态移除图片方法
  • C#实现动态生成表格的方法
  • c# 动态添加非标准html控件的代码
  • C#中事件的动态调用实现方法
  • 使用 C# 动态编译代码和执行的代码
  • c# DataGridView动态添加新行的二个方法
  • c#动态加载卸载DLL的方法
  • C#在运行时动态创建类型的实现方法
  • c#动态编译执行对象方法示例 运用映射机制创建对象
  • c#动态改变webservice的url访问地址
  • C#中动态显示当前系统时间的实例方法
  • c# asp .net 动态创建sql数据库表的方法
  • c# 动态调用webservice的小例子
  • C#中DropdownList动态绑定的两种方法与实现代码
  • c#基础 动态打开,显示,保存,另存为 图片
  • C# 动态调用webService的方法介绍
  • c#动态调用Webservice的两种方法实例
  • php开源软件 iis7站长之家
  • c#开发的程序安装时动态指定windows服务名称
  • C#动态创建组件、属性及事件方法详解
  • 请教,如何动态创建组件对象。送分!
  • 如何使JPanel中的一个继承JPanel的组件的尺寸发生动态改变???
  • JPanel中如何动态生成组件?
  • Applet中在Panel上动态增加组件,怎样立即显示出来?
  • Oracle组件实现动态Web数据库
  • Android实现动态切换组件背景的方法
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • 为什么我动态的写了一个属性文件之后,读出来的还是原来的属性文件呢?
  • ExtJS动态设置TextField的readOnly属性的例子
  • jquery动态改变onclick属性导致失效的问题解决方法
  • Python实现动态添加类的属性或成员函数的解决方法
  • jquery动态改变form属性提交表单
  • 用CSS动态控制文本属性
  • Linux下指定运行时加载动态库路径及shell下执行程序默认路径
  • 可执行程序加载动态库 和 动态库加载动态库 的动态库初始化的问题
  • Linux下c函数dlopen实现加载动态库so文件代码举例
  • 动态库调用动态库
  • linux动态链接库里全局变量和静态变量会导致动态库不可重入吗?
  • linux能否成为动态域名解析客户端的动态域名解析服务器?
  • 关于动态菜单项,注意,不是动态菜单(一定给分)
  • Linux下,怎么确定动态库的绝对位置(动态库里面确定)?
  • 为什么动态生成的图片,JSP不能动态的获得
  • 请问:能不能动态的运行时生成对象(动态定义对象), 要涉及到哪些技术, 反射,序列化...??
  • 动态加载动态库问题--在线等
  • 静态库,动态库,静态链接和动态链接
  • 在linux下(c++),如何动态调用 动态库(.so)? 为什么我编译报错:undefined reference to "dlopen"
  • 程序发布的问题(动态连接库的问题)
  • JSP动态缓存 JimStoneCache
  • 怎么调试unix动态库.
  • 如何调用动态
  • 动态编译与静态编译驱动程式疑问?
  • 关于动态库的问题,谢谢!
  • linux动态路由问题
  • 内核与动态(共享)库


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3