F#中的C#对象初始化语法

时间:2014-05-05 02:10:39

标签: c# f# c#-to-f# object-initializers

请注意:此问题this问题相同。

我最近遇到过一些我以前遇到的C#语法:

有没有办法在F#中做到这一点?

class Two
{
    public string Test { get; set; }
}

class One
{
    public One()
    {
        TwoProperty = new Two();
    }
    public Two TwoProperty { get; private set; }
}

var test = new One(){ TwoProperty = { Test = "Test String" }};

(注意当setter是私有的时初始化器中TwoProperty的初始化 - 它正在TwoProperty中存储的对象上设置属性,但是 NOT 存储一个新属性属性中Two的实例)

编辑: 我最近在monotouch的构造函数中遇到了一些C#代码,如下所示:

nameLabel = new UILabel {
    TextColor = UIColor.White,
    Layer = {
        ShadowRadius = 3,
        ShadowColor = UIColor.Black.CGColor,
        ShadowOffset = new System.Drawing.SizeF(0,1f),
        ShadowOpacity = .5f
    }
};

我能想到的最好的F#翻译是这样的:

let nameLabel = new UILabel ( TextColor = UIColor.White )
do let layer = nameLabel.Layer
   layer.ShadowRadius <- 3.0f
   layer.ShadowColor <- UIColor.Black.CGColor
   layer.ShadowOffset <- new System.Drawing.SizeF(0.0f,1.0f)
   layer.ShadowOpacity <- 0.5f  

这并不可怕,但是重复的layer引用确实会产生更多的噪音,加上它更具说明性和声明性更强。

3 个答案:

答案 0 :(得分:6)

我认为F#不允许在初始化期间设置嵌套属性。假设您是API的作者,解决方法是将整个对象传递给构造函数。这消除了对具有不同可访问性的getter和setter的需求,并使整个F#代码更加清晰。

type Two() =
    member val Test = "" with get, set

type One(twoProperty) =
    member val TwoProperty = twoProperty

let test = One(Two(Test="foo"))

正如您在your comment中提到的,您可以创建一个辅助函数,接受各种属性作为可选参数。类型扩展适用于此:

type UILayer with
    member this.Configure(?shadowRadius, ?shadowColor, ?shadowOffset, ?shadowOpacity) = 
        this.ShadowRadius <- defaultArg shadowRadius this.ShadowRadius
        this.ShadowColor <- defaultArg shadowColor this.ShadowColor
        this.ShadowOffset <- defaultArg shadowOffset this.ShadowOffset
        this.ShadowOpacity <- defaultArg shadowOpacity this.ShadowOpacity

let nameLabel = UILabel(TextColor=UIColor.White)
nameLabel.Layer.Configure(
    shadowRadius=3.0f, 
    shadowColor=UIColor.Black.CGColor, 
    shadowOffset=SizeF(0.0f, 1.0f), 
    shadowOpacity=0.5f)

答案 1 :(得分:3)

将构造封装到单独的初始化函数中是否有意义?

let layerInit layer radius color offset opacity =
    do
       layer.ShadowRadius <- radius
       layer.ShadowColor <- color
       layer.ShadowOffset <- offset
       layer.ShadowOpacity <- opacity
    layer // I do this in case you want to use this fluently or pass in new Layer()

然后在您的代码中使用它:

let nameLabel = new UILabel ( TextColor = UIColor.White )
layerInit nameLabel.Layer 3.0f UIColor.Black.CGColor new System.Drawing.SizeF(0.0f,1.0f) 0.5f |> ignore

答案 2 :(得分:1)

F#4.0推出了一个功能,可能有助于封装@plinth的答案。它允许创建扩展属性,可以在构造函数中初始化。