在F#中命名元组/匿名类型?

Li *_*oyi 34 c# f#

在C#中,您可以执行以下操作:

var a = new {name = "cow", sound = "moooo", omg = "wtfbbq"};
Run Code Online (Sandbox Code Playgroud)

在Python中你可以做类似的事情

a = t(name = "cow", sound = "moooo", omg = "wtfbbq")
Run Code Online (Sandbox Code Playgroud)

当然不是默认情况下,但实现一个t允许你这样做的类是微不足道的.事实上,当我使用Python时,我确实做到了这一点,并且发现它非常方便小型一次性容器,您希望能够通过名称而不是索引(这很容易混淆)来访问组件.

除了这些细节之外,它们与它们所服务的利基中的元组基本相同.

特别是,我现在正在看这个C#代码:

routes.MapRoute(
            "Default", // Route name
            "{controller}/{action}/{id}", // URL with parameters
            new { controller = "Home", action = "Index", id = UrlParameter.Optional } // Parameter defaults
        );
Run Code Online (Sandbox Code Playgroud)

它是F#的等价物

type Route = { 
    controller : string
    action : string
    id : UrlParameter }

routes.MapRoute(
    "Default", // Route name
    "{controller}/{action}/{id}", // URL with parameters
    { controller = "Home"; action = "Index"; id = UrlParameter.Optional } // Parameter defaults
  )
Run Code Online (Sandbox Code Playgroud)

这既冗长又重复,更不用说相当烦人了.你有多接近F#中的这种语法?我现在不介意跳过一些篮球(甚至是火红的篮球!),如果这意味着它会给我一些有用的东西来干掉像这样的代码.

Dan*_*iel 24

我发现它更容易

let route = routes.MapRoute(
    "Default", // Route name
    "{controller}/{action}/{id}" // URL with parameters
    )
route.Defaults.Add("controller", "Home")
route.Defaults.Add("action", "Index")
Run Code Online (Sandbox Code Playgroud)

要么

[ "controller", "Home"
  "action", "Index" ]
|> List.iter route.Defaults.Add
Run Code Online (Sandbox Code Playgroud)

在F#中,我会避​​免调用接受匿名类型的重载,因为我会避免调用接受FSharpListC#的F#方法.这些是特定于语言的功能.通常可以使用与语言无关的重载/解决方法.

编辑

只看文档 - 这是另一种方法

let inline (=>) a b = a, box b

let defaults = dict [
  "controller" => "Home"
  "action"     => "Index" 
]
route.Defaults <- RouteValueDictionary(defaults)
Run Code Online (Sandbox Code Playgroud)


Tom*_*cek 18

你不能在F#中创建"匿名记录" - 使用类型时,你可以使用匿名的元组,但不带标签,或者你可以使用必须事先声明并有标签的记录:

// Creating an anonymous tuple
let route = ("Home", "Index", UrlParameter.Optional)

// Declaration and creating of a record with named fields
type Route = { controller : string; action : string; id : UrlParameter } 
let route = { controller = "Home"; action = "Index"; id = UrlParameter.Optional } 
Run Code Online (Sandbox Code Playgroud)

从技术上讲,匿名记录的问题是它们必须在某处定义为实际类(.NET运行时需要一个类型),但如果编译器将它们放在每个程序集中,那么具有相同成员的两个匿名记录可能是不同的类型如果它们是在不同的程序集中定义的

老实说,我认为您发布的示例只是ASP.NET中一个糟糕的设计决策 - 它滥用特定的C#功能来做一些它没有设计的东西.它可能不是一样糟糕,但它仍然是奇数.该库采用C#匿名类型,但它将其用作字典(即它使用它作为创建键值对的好方法,因为您需要指定的属性是动态的).

因此,如果你正在使用F#中的ASP.NET,那么在不必创建记录的情况下使用替代方法可能更容易 - 如果ASP.NET API提供了一些替代方法(如Daniel所示,那就更好了)写的方式).

  • @Kirk - 即使在C#中,这种方法仍然存在问题,例如可发现性.`defaults`参数的类型为`object`,描述为"包含默认路由值的对象".作为一个调用者,这是完全不可思议的,除非我已经看到了库使用的非常惯用的样式的例子.对我来说,这是C#无法支持简洁的集合文字的解决方法的症状,而不是积极使用F#缺乏的C#特性. (7认同)
  • 关键是,如果ASP.NET是一个.NET库(而不是一个C#库),它就不应该对人们将要使用的语言做出任何假设.从纯.NET的角度来看,使用具有属性来表示字典的类听起来非常愚蠢! (6认同)
  • 在C#中我不会说'IDictionary`更清晰或更短,但考虑到几种语言针对CLR,它是一种更好的方法(快乐的媒介).匿名类型特定于C#. (3认同)

Tah*_*san 11

OP没有描述匿名类型的最佳使用.当使用LINQ映射到任意类时,最好使用它们.例如:

var results = context.Students
              .Where(x => x.CourseID = 12)
              .Select(x => new { 
                 StudentID = x.ID, 
                 Name = x.Forename + " " + x.Surname
              });
Run Code Online (Sandbox Code Playgroud)

我知道这可以通过定义一个新的记录类型来完成,但是你有两个地方来维护代码,(1)你已经使用它的记录类型定义(2).

它可以用元组来完成,但要访问各个字段,你必须(studentId, name)始终使用解构语法.如果元组中有5个项目,这将变得难以处理.我宁愿输入x并点击dot并让intellisense告诉我哪些字段可用.


Ton*_*ony 6

现在在F#4.6(预览版)中,我们有匿名记录

因此,我们可以使用以下代码语法:

let AwesomeAnonymous = {|  ID = Guid.NewGuid()
                           Name = "F#"
                        |}

AwesomeAnonymous.Name |> Debug.WriteLine
Run Code Online (Sandbox Code Playgroud)

Visual Studio Intellisense也支持它: 屏幕截图

这样的代码可能是这样的:

routes.MapRoute(
    "Default", // Route name
    "{controller}/{action}/{id}", // URL with parameters
    {| controller = "Home"; action = "Index"; id = UrlParameter.Optional |} // Parameter defaults
  )
Run Code Online (Sandbox Code Playgroud)

另请参阅:宣布F#4.6预览