使用 GetType().Name 在事件处理程序中强制转换发送者对象

发布于 2024-07-15 01:42:43 字数 421 浏览 14 评论 0原文

我有一个 Textbox 和 RichTextBox 的事件处理程序。 代码是相同的,但

在处理程序 #1 中我这样做:

RichTextBox tb = (RichTextBox)sender

在处理程序 #2 中相应地:

TextBox tb = (TextBox)sender

这样做我可以完全操纵发送控件。 如何根据使用的类型将发送对象转换为 TextboxRichTextbox

sender.GetType().Name

,然后在运行时创建控件并使用它? 这样我只需要一个事件处理函数:更少的代码,更少的错误,更容易维护并且干燥:-)

I have an event handler for a Textbox and a RichTextBox.
The code is identical, but

In handler #1 I do:

RichTextBox tb = (RichTextBox)sender

In handler #2 accordingly:

TextBox tb = (TextBox)sender

Doing so I can fully manipulate the sending control.
How can I cast the sending object to Textbox or RichTextbox according to its type using

sender.GetType().Name

and then create the control at runtime and work with it? That way I only need one event handler function: less code, less errors, easier to maintain and DRY :-)

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(10

〃温暖了心ぐ 2024-07-22 01:42:43

你永远不必施放。 当我开始时,我也曾以同样的方式思考过,这种“模式”是不正确的,而且不符合逻辑。

你最好的选择是使用类似的东西:

if (sender is TextBox)
{
  TextBox tb = (TextBox)sender;
}
else if (sender is RichTextBox)
{
  RichTextBox rtb = (RichTextBox)sender;
}
else
{
  // etc
}

You never have to cast. I used to think the same way when I started, this 'pattern' is incorrect, and not really logical.

Your best bet is to use something like:

if (sender is TextBox)
{
  TextBox tb = (TextBox)sender;
}
else if (sender is RichTextBox)
{
  RichTextBox rtb = (RichTextBox)sender;
}
else
{
  // etc
}
甜妞爱困 2024-07-22 01:42:43

我知道这是一篇非常古老的文章,但在 Framework 4 中,您可以将发送者转换为控件:

Control cntrl = (Control)sender;
cntrl.Text = "This is a " + sender.GetType().ToString();

请注意,您只能引用所有不同控件共有的控件(即文本)。

I know this is a very old post but in Framework 4 you can cast the sender as a Control:

Control cntrl = (Control)sender;
cntrl.Text = "This is a " + sender.GetType().ToString();

Note you are only able to reference controls that all of the different controls have in common (ie Text).

森林散布 2024-07-22 01:42:43

您可以使用 'is'。

如果您只想知道类型并且不需要对象引用:

if (sender is RichTextBox)
{
    // ...
}
else if (sender is TextBox)
{
    // ...
}

但是您通常确实需要对象:C#7 有一个很好的语法,允许您测试并获取内联值:

if (sender is RichTextBox richTextBox)
{
    richTextBox.Text = "I am rich";
}
else if (sender is TextBox textBox)
{
    textBox.Text = "I am not rich";
}

Rather than the type name you could use 'is'.

If you just want to know the type and don't need an object reference:

if (sender is RichTextBox)
{
    // ...
}
else if (sender is TextBox)
{
    // ...
}

However you generally do want the object: C#7 has a nice syntax that allows you to test and get the value inline:

if (sender is RichTextBox richTextBox)
{
    richTextBox.Text = "I am rich";
}
else if (sender is TextBox textBox)
{
    textBox.Text = "I am not rich";
}
以歌曲疗慰 2024-07-22 01:42:43
RichTextBox textbox = sender as RichTextBox;
if (textbox != null)
{
   // do stuff as a rtb
   textbox.Text = "I'm a rtb";
   return;
}

TextBox textbox = sender as TextBox;
if (textbox != null)
{
   // do stuff as a textbox
   textbox.Text = "I'm a textbox";
}
RichTextBox textbox = sender as RichTextBox;
if (textbox != null)
{
   // do stuff as a rtb
   textbox.Text = "I'm a rtb";
   return;
}

TextBox textbox = sender as TextBox;
if (textbox != null)
{
   // do stuff as a textbox
   textbox.Text = "I'm a textbox";
}
↘紸啶 2024-07-22 01:42:43

根据您需要的属性,您可以将发送者强制转换为 TextBoxBase,因为 TextBox 和 RichTextBox 都继承自该子类。

Depending on what properties you need, you could cast the sender as a TextBoxBase as both the TextBox and RichTextBox both inherit from that sub-class.

爱的故事 2024-07-22 01:42:43

转换只能在编译时完成,因此您需要知道您希望在编译时转换为的类型。 因此,在转换时不能使用运行时类型(由 GetType() 返回)。

如果您正在寻找多态性,您可以通过反射访问 Name 属性。 我不会只是为了能够重用事件处理程序而这样做。

如果您想要强类型,两个发送者上的公共基类或接口是唯一的方法。

Casting can only be done at compile-time and thus you need to know the types that you wish to cast to at compile-time. A runtime Type (as returned by GetType()) can therefore not be used when casting.

If it is polymorphism you are looking for you could access the Name property through reflection. I wouldn't go that way though just to be able to reuse event handlers.

If you want strong typing, a common base class or interface on the two senders is the only way to go.

Oo萌小芽oO 2024-07-22 01:42:43

上述代码的通用版本:

public static void CastAndUse<T>(object item, Action<T> action) where T : class
{
    T thing = item as T;

    if (thing != null)
    {
        action(thing);
    }
}

用作:

CastAndUse(sender, new Action((foo) => foo = bar));

不完美,但方便。

Generic version of the above code:

public static void CastAndUse<T>(object item, Action<T> action) where T : class
{
    T thing = item as T;

    if (thing != null)
    {
        action(thing);
    }
}

Used as:

CastAndUse(sender, new Action((foo) => foo = bar));

Not perfect, but handy.

相对绾红妆 2024-07-22 01:42:43

您还可以使用内联临时变量来处理转换。

if (sender is RichTextBox tb)
{
    // ... //
} 
else if (sender is TextBox tb)
{
    // ... //
}

You can also use an inline-temporary variable to handle the cast for you.

if (sender is RichTextBox tb)
{
    // ... //
} 
else if (sender is TextBox tb)
{
    // ... //
}
独闯女儿国 2024-07-22 01:42:43

如果代码相同,您需要关心吗? 我想知道转换为 Control 是否无法为您提供所需的一切...

一个复杂的处理程序不一定比几个简单的处理程序更好。 无论哪种方式,如果您必须走这条路,“as”/“is”是更好的选择(它不依赖于字符串等):

TextBox tb = sender as TextBox;
if(tb!=null) {/* TextBox specific code */}
...

If the code is identical, do you need to care? I wonder if casting to Control wouldn't give you everything you need...

One complex handler is not necessarily better than several simple handlers. Either way, if you have to go this route, "as"/"is" is preferable (it isn't dependent on strings etc):

TextBox tb = sender as TextBox;
if(tb!=null) {/* TextBox specific code */}
...
静谧 2024-07-22 01:42:43

如果您不想重复代码,那么您可以强制转换这两个控件,将常见操作重构为以 TextBoxBase 作为参数的单独方法。 在事件处理程序中,将控件转换为 System.Windows.Forms.TextBoxBase,因为这两个控件均派生自 TexbBoxBase 并调用该方法。

请注意,如果您需要任何这些控件的特定属性,那么此重构将不起作用。

if you dont want to repeat the code then you can cast both the controls, refactor the common actions to a separate method which takes TextBoxBase as an argument. And in your event handlers convert the controls to System.Windows.Forms.TextBoxBase as both controls are derived from the TexbBoxBase and call the method.

Please note If you need specific properties of any of these controls then this refactoring wont work.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文