F#将带参数的运算符传递给函数

dan*_*dan 10 f# operators partial-application

你可以使用部分应用的运算符传递"除以2"或"减1"等操作,其中"加1"如下所示:

List.map ((+) 1) [1..5];;  //equals [2..6]
// instead of having to write: List.map (fun x-> x+1) [1..5]
Run Code Online (Sandbox Code Playgroud)

正在发生的事情是1被应用于(+)作为它的第一个参数,并且列表项被应用为第二个参数.对于加法和乘法,这个参数排序无关紧要.

假设我想从每个元素中减去1(这可能是常见的初学者错误):

List.map ((-) 1) [1..5];;  //equals [0 .. -4], the opposite of what we wanted
Run Code Online (Sandbox Code Playgroud)

1被应用于( - )作为它的第一个参数,所以不是(list_item - 1),我得到(1 - list_item).我可以将其重写为添加负数而不是减去正数:

List.map ((+) -1) [1..5];;
List.map (fun x -> x-1) [1..5];; // this works too
Run Code Online (Sandbox Code Playgroud)

我正在寻找一种更具表现力的方式来编写它,例如((-) _ 1),where _表示占位符,就像在Arc语言中一样.这将导致1成为第二个参数-,因此在List.map中,它将评估为list_item - 1.所以如果你想映射divide by 2到列表,你可以写:

List.map ((/) _ 2) [2;4;6] //not real syntax, but would equal [1;2;3] 
List.map (fun x -> x/2) [2;4;6] //real syntax equivalent of the above
Run Code Online (Sandbox Code Playgroud)

可以这样做还是我必须使用(fun x -> x/2)?似乎我们最接近占位符语法的方法是使用带有命名参数的lambda.

Log*_*ldo 14

你可以编写一个翻转函数,例如:

let flip f x y = f y x

List.map (flip (-) 1) [2;4;6]
Run Code Online (Sandbox Code Playgroud)

我可能有错误的语法,我对F#不是很精通.

  • 您的翻转功能(也可以)被定义为'let flip fxy = fyx' (8认同)

Bri*_*ian 10

F#,la Haskell中没有"操作部分",也没有占位符参数(显然是la Arc).您可以使用另一个答案中建议的"翻转"组合器来反转参数的顺序,然后部分应用第一个(现在是第二个)参数.

但我会用

fun x -> x / 2
Run Code Online (Sandbox Code Playgroud)

除非你正在玩代码高尔夫,否则我不认为在这里试图刮掉另外几个角色会给你带来任何好处.


Joh*_*bom 9

flip所建议-溶液洛根CAPALDO也可以使用运营商(这里写的>.):

let (>.) x f = (fun y -> f y x)
List.map (1 >. (-)) [2;4;6]
Run Code Online (Sandbox Code Playgroud)

或者如果你更喜欢操作数:

let (>.) f x = (fun y -> f y x)
List.map ((-) >. 1) [2;4;6]
Run Code Online (Sandbox Code Playgroud)

编辑:使用"看起来更像占位符"的运算符(这里>-<)让您非常接近建议的语法:

List.map ((-) >-< 1) [2;4;6]
Run Code Online (Sandbox Code Playgroud)

'_'不幸(?)不是F#中的有效运算符符号.