我可以使用F#中的管道运算符将参数传递给构造函数吗?

时间:2010-11-06 03:39:52

标签: f#

此代码反转字符串:

let reverse (s : string) = new string(s.ToCharArray() |> Array.rev)

可以使用管道运算符重写它以将所需的参数传递给string()构造函数吗?例如,这似乎更惯用:

// Doesn't compile:
let reverse (s : string) = s.ToCharArray() |> Array.rev |> new string

同样,为什么我不能以下列方式使用string运算符?

let reverse2 (s : string) = s.ToCharArray() |> Array.rev |> string

这是在行动:

> reverse2 "foo" ;;
val it : string = "System.Char[]"

它返回的类型而不是“oof”。

2 个答案:

答案 0 :(得分:9)

不,管道运算符只能与F#函数一起使用,它不能与类构造函数,成员方法或静态方法一起使用。原因是这些方法支持的重载会使F#的类型推断复杂化。但是,如果您真的想使用管道,可以将char数组的每个元素映射到String,然后将该序列传递到Seq.concat ""

s.ToCharArray() |> Array.rev |> Seq.map(string) |> String.concat ""

或者你可以用F#方法包装字符串构造函数调用:

let stringCArrCtor (carr: char[]) =
    new string(carr)

s.ToCharArray() |> Array.rev |> stringCArrCtor

回答你的上一个问题,

s.ToCharArray() |> Array.rev |> string

无法使用,因为它等同于

(s.ToCharArray() |> Array.rev).ToString()

并且不会覆盖Array ToString()方法,因此它只返回默认的反射类型名称。

答案 1 :(得分:9)

正如斯蒂芬所提到的,最好的办法是定义一个调用构造函数的新函数。您可以将它放入名为String的模块中(在某些命名空间中),这样您就可以获得与使用其他F#函数时类似的感觉。我可能会用:

module String =
  let create (c:char[]) = new string(c)

使用构造函数作为一等价值的问题出现在SO之前,但我再也找不到我之前的答案 - 有一个非常疯狂的技巧可以提供你的能力,但它是一个巨大的黑客(没有人应该使用它和F#的下一个版本是希望不会允许的。无论如何,您可以使用statically resolved type parameters来编写以下内容:

let inline ctor< ^R, ^T when ^R : 
        (static member ``.ctor`` : ^T -> ^R)> (arg:^T) =
    (^R : (static member ``.ctor`` : ^T -> ^R) arg)

并使用这样的功能:

"ABC".ToCharArray() |> Array.rev |> ctor<string, _>;;

ctor函数基本上要求指定为第一个类型参数的类型具有构造函数,并且它调用构造函数(另一个类型参数是构造函数的参数,并由编译器推断)。但这实际上只是一种好奇心 - 定义自己的功能是最好的方法。