- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我想知道编译器/运行时如何确定 lambda 表达式的类型?
例如下面的System.Linq
Select
扩展方法(非select查询)
//var recordIds = new List<int>(records.Select(r => r?.RecordId ?? 0));
//var recordIds = new List<int>(records.Where(r => r != null).Select(r => r.RecordId));
var recordIds = new List<int>(records.Select(r => r.RecordId));
定义为
Enumerable.Select<TSource, TResult> Method (IEnumerable<TSource>, Func<TSource, TResult>)
因此采用 lambda r => r.RecordId
作为Func<TSource, TResult>
.
lambda 的类型是如何确定的,一旦确定,是否只是简单地转换为该类型?
最佳答案
I was wondering how the the compiler/runtime determines a lambda expression's type?
比较复杂。实现此功能是发布 C# 3 的 Visual Studio 版本的“长杆”——因此,我在这方面超过计划的每一天都是 VS 失误的一天! -- 随着在 C# 4 中引入协变和逆变,该功能只会变得更加复杂。
正如其他人所指出的,请查阅规范以了解确切的详细信息。您也可以阅读我多年来撰写的有关它的各种文章。
虽然我可以给你一个快速的概述。假设我们有
records.Select(r => r.RecordId)
哪里records
类型为 IEnumerable<Record>
和 RecordId
类型为 int
.
第一个问题是“IEnumerable<Record>
是否有任何适用的方法称为 Select
?不,没有。因此我们进入扩展方法回合。
那么第二个问题是:“是否有任何具有适用于此调用的可访问扩展方法的静态类型?”
SomeType.Select(records, r => r.RecordId)
让我们假设 Enumerable
是唯一的这种类型。它有两个版本的 Select
,其中一个采用带有两个参数的 lambda。我们可以自动丢弃那个。正如您所指出的,这给我们留下了:
static IEnumerable<R> Select<A, R>(IEnumerable<A>, Func<A, R>)
第三个问题:能否推导出类型参数对应的类型实参A
和 R
?
在第一轮类型推断中,我们考虑了所有非 lambda 参数。我们只有一个。我们推断 A
可能是 Record
.然而,IEnumerable<T>
是协变的,所以我们注意到它可能是比 Record
更通用的类型以及。它不能是比 Record
更具体的类型虽然。
现在我们问“我们完成了吗?”不,我们仍然不知道R
.
“还有什么可以推断的吗?”是的。我们还没有检查 lambda。
“关于 A
是否有任何矛盾或额外的事实需要了解?”没有。
因此我们“修复”A
至 Record
继续前进。到目前为止我们知道什么?我们有:
static IEnumerable<R> Select<Record, R>(IEnumerable<Record>, Func<Record, R>)
然后我们说 OK,参数必须是 (Record r) => r.RecordId
.我们可以推断出这个 lambda 的返回类型吗?显然是的,它是 int
.所以我们在 R
上写了一个注释说它可能是int
.
我们完成了吗?是的。还有什么我们可以推论的吗?不,我们是否推断出所有的类型参数?是的。所以我们“修复”R
至 int
,我们就完成了。
现在我们进行最后一轮检查以确保 Select<Record, int>
不会产生任何错误;例如,如果 Select
有一个被 <Record, int>
违反的通用约束我们现在会拒绝它。
我们推断出 records.Select(r=>r.RecordId)
与 Enumerable.Select<Record, int>(records, (Record r) => { return (int)r.RecordId; } )
含义相同这是对该方法的合法调用,所以我们完成了。
当您引入多个边界时,事情会变得更加复杂。看看你能不能算出像 Join
这样的东西在有四种类型参数要推断的情况下工作。
关于c# - 编译器/运行时如何确定 lambda 表达式的类型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49677164/
可以使用 lambda 和函数创建有序对(Lisp 中的缺点),如 Use of lambda for cons/car/cdr definition in SICP 所示。 它也适用于 Python
我正在尝试从另一个调用一个 AWS lambda 并执行 lambda 链接。这样做的理由是 AWS 不提供来自同一个 S3 存储桶的多个触发器。 我创建了一个带有 s3 触发器的 lambda。第一
根据以下源代码,常规 lambda 似乎可以与扩展 lambda 互换。 fun main(args: Array) { val numbers = listOf(1, 2, 3) f
A Tutorial Introduction to the Lambda Calculus 本文介绍乘法函数 The multiplication of two numbers x and y ca
我想弄清楚如何为下面的表达式绘制语法树。首先,这究竟是如何表现的?看样子是以1和2为参数,如果n是 0,它只会返回 m . 另外,有人可以指出解析树的开始,还是一个例子?我一直找不到一个。 最佳答案
在 C++0x 中,我想知道 lambda 函数的类型是什么。具体来说: #include type1 foo(int x){ return [x](int y)->int{return x * y
我在其中一个职位发布中看到了这个问题,它询问什么是 lambda 函数以及它与高阶函数的关系。我已经知道如何使用 lambda 函数,但不太自信地解释它,所以我做了一点谷歌搜索,发现了这个:What
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
Evaluate (((lambda(x y) (lambda (x) (* x y))) 5 6) 10) in Scheme. 我不知道实际上该怎么做! ((lambda (x y) (+ x x
我正在处理 MyCustomType 的实例集合如下: fun runAll(vararg commands: MyCustomType){ commands.forEach { it.myM
Brian 在他对问题 "Are side effects a good thing?" 的论证中的前提很有趣: computers are von-Neumann machines that are
在 Common Lisp 中,如果我希望两个函数共享状态,我将按如下方式执行 let over lambda: (let ((state 1)) (defun inc-state () (in
Evaluate (((lambda(x y) (lambda (x) (* x y))) 5 6) 10) in Scheme. 我不知道实际上该怎么做! ((lambda (x y) (+ x x
作为lambda calculus wiki说: There are several possible ways to define the natural numbers in lambda cal
我有一个数据类,我需要初始化一些 List .我需要获取 JsonArray 的值(我使用的是 Gson)。 我做了这个函数: private fun arrayToList(data: JsonAr
((lambda () )) 的方案中是否有简写 例如,代替 ((lambda () (define x 1) (display x))) 我希望能够做类似的事情 (empty-lam
我在 Java library 中有以下方法: public void setColumnComparator(final int columnIndex, final Comparator colu
我正在研究一个函数来计算国际象棋游戏中棋子的有效移动。 white-pawn-move 函数有效。当我试图将其概括为任一玩家的棋子 (pawn-move) 时,我遇到了非法函数调用。我已经在 repl
考虑这段代码(在 GCC 和 MSVC 上编译): int main() { auto foo = [](auto p){ typedef decltype(p) p_t;
我正在阅读一个在 lambda 内部使用 lambda 的片段,然后我想通过创建一个虚拟函数来测试它,该函数从文件中读取然后返回最大和最小数字。 这是我想出来的 dummy = lambda path
我是一名优秀的程序员,十分优秀!