作为第一个参数委托给扩展方法

2023-12-26

女士们先生们,

我最近尝试了这个实验:

static class TryParseExtensions
{
    public delegate bool TryParseMethod<T>(string s, out T maybeValue);
    public static T? OrNull<T>(this TryParseMethod<T> tryParser, string s) where T:struct 
    {
        T result;
        return tryParser(s, out result) ? (T?)result : null;
    }
}

// compiler error "'int.TryParse(string, out int)' is a 'method', which is not valid in the given context"
var result = int.TryParse.OrNull("1");  // int.TryParse.OrNull<int>("1"); doesnt work either

// compiler error: type cannot be infered....why?
var result2 = TryParseExtensions.OrNull(int.TryParse, "2"); 

// works as expected
var result3 = TryParseExtensions.OrNull<int>(int.TryParse, "3");
      var result4 = ((TryParseExtensions.TryParseMethod<int>)int.TryParse).OrNull("4");

我想知道两件事:

  • 为什么编译器无法推断出“int”类型参数?

  • 我是否正确理解扩展方法不会在委托类型上被发现,因为我猜它们并不是真正的那种类型(而是一个“方法”),只是恰好与委托签名匹配?这样的演员阵容解决了这个问题。场景 1 是否可行(当然不是具体的场景,而是一般情况)?我想从语言/编译器的角度来看,它实际上有用,还是我只是(试图)在这里疯狂地滥用东西?

期待一些见解。谢谢


您有很多问题。 (将来我建议当您有多个问题时,将它们分成多个问题,而不是在一张帖子中包含多个问题;您可能会得到更好的答复。)

为什么编译器无法推断出以下中的“int”类型参数:

TryParseExtensions.OrNull(int.TryParse, "2");  

好问题。我不在这里回答这个问题,而是向您推荐我 2007 年的文章,其中解释了为什么这在 C# 3.0 中不起作用:

http://blogs.msdn.com/b/ericlippert/archive/2007/11/05/c-3-0-return-type-in​​ference-does-not-work-on-member-groups.aspx http://blogs.msdn.com/b/ericlippert/archive/2007/11/05/c-3-0-return-type-inference-does-not-work-on-member-groups.aspx

总结:从根本上来说,这里存在一个先有鸡还是先有蛋的问题。我们必须对 int.TryParse 进行重载解析,以确定 TryParse 的哪个重载是预期的重载(或者,如果它们都不起作用,则错误是什么。)重载解析总是尝试从论点。但在这种情况下,这正是我们试图推断的论证类型。

我们可以提出一种新的重载解析算法,它说“好吧,如果方法组中只有一个方法,那么即使我们不知道参数是什么,也选择那个方法”,但这似乎很弱。对于只有一个方法的特殊情况方法组来说,这似乎是一个坏主意,因为这会因为添加新的重载而受到惩罚;它可能突然成为一个重大变化。

正如您从该文章的评论中看到的,我们收到了很多很好的反馈。得到的最好的反馈基本上是“好吧,假设类型推断有already计算出所有参数的类型,它是返回类型我们正在尝试推断;在这种情况下,您可以进行重载解析”。该分析是正确的,并且对该效果的更改进入了 C# 4。我在这里详细讨论了这一点:

http://blogs.msdn.com/b/ericlippert/archive/2008/05/28/method-type-in​​ference-changes-part-zero.aspx http://blogs.msdn.com/b/ericlippert/archive/2008/05/28/method-type-inference-changes-part-zero.aspx

我是否正确理解扩展方法不会在委托类型上被发现,因为我猜它们并不是真正的那种类型(而是一个“方法”),只是恰好与委托签名匹配?

你的术语有点偏离,但你的想法是正确的。当“接收者”是一个时,我们不会发现扩展方法方法组。更一般地说,当接收者缺乏自己的类型时,我们不会发现扩展方法,而是根据其上下文采用类型:方法组、lambda、匿名方法和 null 文字都具有此属性。说起来真的很奇怪null.Whatever()并让其调用 String 上的扩展方法,甚至更奇怪,(x=>x+1).Whatever()并让其调用扩展方法Func<int, int>.

描述此行为的规范行是:

从[接收者表达式]到第一个参数的类型[...]的隐式标识、引用或装箱转换[必须存在]。

方法组上的转换不是恒等、引用或装箱转换;它们是方法组转换。

场景 1 是否可行(当然不是具体的场景,而是一般情况)?我想从语言/编译器的角度来看,它实际上有用,还是我只是(试图)在这里疯狂地滥用东西?

它不是不可行的。我们这里有一支非常聪明的团队,但没有理论的不可能这样做的原因。在我们看来,它并不是一个为语言增加更多价值的功能,而不是额外的复杂性成本。

有时它会很有用。例如,我希望能够做到这一点;假设我有一个static Func<A, R> Memoize<A, R>(this Func<A, R> f) {...}:

var fib = (n=>n<2?1:fib(n-1)+fib(n-2)).Memoize();

而不是你今天必须写的内容是:

Func<int, int> fib = null;
fib = n=>n<2?1:fib(n-1)+fib(n-2);
fib = fib.Memoize();

但坦率地说,所提议的功能给语言增加的额外复杂性并没有因为使上面的代码变得不那么冗长而带来的小小的好处而得到补偿。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

作为第一个参数委托给扩展方法 的相关文章

  • 在 LINQ 查询中返回不带时间的日期

    我正在编写一个查询 我想计算按日期联系我们的呼叫中心的次数 看起来很简单 但由于联系日期字段是日期时间字段 我得到了时间 因此当我按联系日期 时间 分组时 每个联系日期实例的计数为 1 所以 我想只按日期分组 而不按时间分组 下面是我用来查
  • 自动从 C# 代码进行调试过程并读取寄存器值

    我正在寻找一种方法来读取某个地址的 edx 注册表 就像这个问题中所问的那样 读取eax寄存器 https stackoverflow com questions 16490906 read eax register 虽然我的解决方案需要用
  • 如何在没有 Control.Invoke() 的情况下从后台线程修改控件属性

    最近 我们遇到了一些旧版 WinForms 应用程序 我们需要更新一些新功能 在专家测试该应用程序时 发现一些旧功能被破坏 无效的跨线程操作 现在 在您认为我是新手之前 我确实有一些 Windows 窗体应用程序的经验 我不是专家 但我认为
  • 为什么禁止在 constexpr 函数中使用 goto?

    C 14 对你能做什么和不能做什么有规则constexpr功能 其中一些 没有asm 没有静态变量 看起来相当合理 但标准也不允许goto in constexpr功能 即使它允许其他控制流机制 这种区别背后的原因是什么 我以为我们已经过去
  • 写入和读取文本文件 - C# Windows 通用平台应用程序 Windows 10

    有用 但在显示任何内容之前 您必须在文本框中输入内容 我想那是因为我使用了 TextChanged 事件处理程序 如果我希望它在没有用户交互的情况下显示文本文件的内容 我应该使用哪个事件处理程序 因此 我想在按下按钮时将一些数据写入 C W
  • HttpClient 像浏览器一样请求

    当我通过 HttpClient 类调用网站 www livescore com 时 我总是收到错误 500 可能服务器阻止了来自 HttpClient 的请求 1 还有其他方法可以从网页获取html吗 2 如何设置标题来获取html内容 当
  • 按字典顺序对整数数组进行排序 C++

    我想按字典顺序对一个大整数数组 例如 100 万个元素 进行排序 Example input 100 21 22 99 1 927 sorted 1 100 21 22 927 99 我用最简单的方法做到了 将所有数字转换为字符串 非常昂贵
  • 为什么模板不能位于外部“C”块内?

    这是一个后续问题一个答案 https stackoverflow com questions 4866433 is it possible to typedef a pointer to extern c function type wit
  • 使用向量的 merge_sort 在少于 9 个输入的情况下效果很好

    不知何故 我使用向量实现了合并排序 问题是 它可以在少于 9 个输入的情况下正常工作 但在有 9 个或更多输入的情况下 它会执行一些我不明白的操作 如下所示 Input 5 4 3 2 1 6 5 4 3 2 1 9 8 7 6 5 4 3
  • 使用安全函数在 C 中将字符串添加到字符串

    我想将文件名复制到字符串并附加 cpt 但我无法使用安全函数 strcat s 来做到这一点 错误 字符串不是空终止的 我确实设置了 0 如何使用安全函数修复此问题 size strlen locatie size nieuw char m
  • C 中的位移位

    如果与有符号整数对应的位模式右移 则 1 vacant bit will be filled by the sign bit 2 vacant bit will be filled by 0 3 The outcome is impleme
  • AccessViolationException 未处理

    我正在尝试使用史蒂夫 桑德森的博客文章 http blog stevensanderson com 2010 01 28 editing a variable length list aspnet mvc 2 style 为了在我的 ASP
  • 什么是 C 语言的高效工作流程? - Makefile + bash脚本

    我正在开发我的第一个项目 该项目将跨越多个 C 文件 对于我的前几个练习程序 我只是在中编写了我的代码main c并使用编译gcc main c o main 当我学习时 这对我有用 现在 我正在独自开展一个更大的项目 我想继续自己进行编译
  • 将应用程序从 Microsoft Access 迁移到 VB 或 C#.NET

    我目前正试图说服管理层需要将我们的应用程序之一移植到 NET 该应用程序已经发展成为 Access 中的一个庞然大物 SQL 后端 拥有 700 个链接表 650 个表单 子表单 130 个模块和 850 个查询 我几乎知道这样做的所有主要
  • 已过时 - OpenCV 的错误模式

    我正在使用 OpenCV 1 进行一些图像处理 并且对 cvSetErrMode 函数 它是 CxCore 的一部分 感到困惑 OpenCV 具有三种错误模式 叶 调用错误处理程序后 程序终止 Parent 程序没有终止 但错误处理程序被调
  • 如何构建印度尼西亚电话号码正则表达式

    这些是一些印度尼西亚的电话号码 08xxxxxxxxx 至少包含 11 个字符长度 08xxxxxxxxxxx 始终以 08 开头 我发现这个很有用 Regex regex new Regex 08 0 9 0 9 0 9 0 9 0 9
  • ListDictionary 类是否有通用替代方案?

    我正在查看一些示例代码 其中他们使用了ListDictionary对象来存储少量数据 大约 5 10 个对象左右 但这个数字可能会随着时间的推移而改变 我使用此类的唯一问题是 与我所做的其他所有事情不同 它不是通用的 这意味着 如果我在这里
  • 在Linux中使用C/C++获取机器序列号和CPU ID

    在Linux系统中如何获取机器序列号和CPU ID 示例代码受到高度赞赏 Here http lxr linux no linux v2 6 39 arch x86 include asm processor h L173Linux 内核似
  • 方法参数内的变量赋值

    我刚刚发现 通过发现错误 你可以这样做 string s 3 int i int TryParse s hello out i returns false 使用赋值的返回值是否合法 Obviously i is but is this th
  • 在 ASP.NET 中将事件冒泡为父级

    我已经说过 ASP NET 中的层次结构 page user control 1 user control 2 control 3 我想要做的是 当控件 3 它可以是任何类型的控件 我一般都想这样做 让用户用它做一些触发回发的事情时 它会向

随机推荐