C#扩展方法(Extension Method)

在 C# 中,扩展方法(Extension Method)的语法允许你通过 this 关键字将第一个参数隐式传递,因此在调用时 ​不需要显式传递第一个参数。

这里有个扩展方法

public static Task<T> InvokeAsync<T>(this Control control, Func<T> func)
{
    var tcs = new TaskCompletionSource<T>();
    control.BeginInvoke(new Action(() =>
    {
        try
        {
            tcs.SetResult(func());
        }
        catch (Exception ex)
        {
            tcs.SetException(ex);
        }
    }));
    return tcs.Task;
}

扩展方法的特点:

**​隐式 control 参数:**
    this Control control 表示这是一个扩展方法,作用在 Control 类型上。
    当你调用 _uiControl.InvokeAsync(showDialogFunc) 时,_uiControl 会自动作为 control 参数传递,而 showDialogFunc 是第二个参数 func。
**​泛型参数 <T>:**
    InvokeAsync<T> 的泛型参数 <T> 会根据 func 的返回值类型自动推断。例如,如果 showDialogFunc 返回 string,则 T 会被推断为 string。

调用时的参数传递:

return await _uiControl.InvokeAsync(showDialogFunc);
_uiControl 是扩展方法的第一个参数 control(通过 this Control control 隐式传递)。
showDialogFunc 是第二个参数 func。

为什么不需要显式传递两个参数?

扩展方法的语法简化了调用方式:
原本需要写为 InvokeAsync(_uiControl, showDialogFunc)。
但因为 this Control control 的存在,你可以直接写成 _uiControl.InvokeAsync(showDialogFunc),编译器会自动将 _uiControl 作为第一个参数。

完整流程解析:

线程切换:
如果当前线程不是 UI 线程,control.BeginInvoke 会将 func 委托的调用调度到 UI 线程。
如果当前已经是 UI 线程,BeginInvoke 会直接执行(同步或异步取决于具体实现)。
​异步包装:
TaskCompletionSource 用于将 BeginInvoke 的异步操作包装成一个 Task。
当 func 在 UI 线程执行完成后:
如果成功,调用 tcs.SetResult(func()) 设置结果。
如果抛出异常,调用 tcs.SetException(ex) 传递异常。
​ 返回 Task:
最终返回的 tcs.Task 可以让调用者通过 await 等待结果。

总结:

​**扩展方法的语法:通过 this 关键字隐式传递第一个参数(control),调用时不需要显式传递。
泛型推断:**根据 func 的返回值类型自动推断泛型参数 。
​线程安全: BeginInvoke 确保 func 在 UI 线程执行,并通过 Task 提供异步支持。

扩展方法Extension Method)是C# 3.0中的一个新特性,它允许程序员向已有的类添加新的方法,而无需修改原始类的定义,也无需继承或实现接口。扩展方法可以为任何类型添加新的方法,包括系统类型,而不需要修改原始类型的代码。 使用扩展方法能够使代码更加简洁、易读,同时也能够提高代码的复用性和可维护性。扩展方法可以应用于许多场景,例如: 1. 在不修改已有类的情况下,为其添加新的功能。 2. 扩展系统类的功能,例如为字符串添加一个新的方法,用于判断字符串是否为空。 3. 扩展第三方库的功能,例如为某个第三方库提供一个新的扩展方法。 下面是一个扩展方法的示例,为字符串类型添加了一个名为IsNullOrEmpty的扩展方法: ```csharp public static class StringExtensions { public static bool IsNullOrEmpty(this string str) { return string.IsNullOrEmpty(str); } } // 使用扩展方法 string myString = null; bool result = myString.IsNullOrEmpty(); ``` 需要注意的是,扩展方法必须声明在一个静态类中,并且必须是一个静态方法扩展方法必须以this关键字作为第一个参数,并且该参数必须是扩展方法所要扩展的类型。本例中,IsNullOrEmpty方法的第一个参数是this string str,即扩展方法所要扩展的类型是字符串类型。 总之,扩展方法C#中一个非常有用的特性,可以为程序员带来便利和效率。在实际开发中,我们需要根据具体情况合理使用扩展方法,以便提高代码的可读性、可重用性和可维护性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值