C#中的模拟方法级联

时间:2013-01-09 04:16:11

标签: c# dart method-cascades

Dart编程语言支持method cascades。方法级联将允许以下Silverlight / WPF C#代码:

var listBox = new ListBox();

listBox.Width = 200;
listBox.MouseEnter += (s, e) => Console.WriteLine("MouseEnter");

var button1 = new Button() { Content = "abc" };
button1.Click += (s, e) => Console.WriteLine("button1.Click");

listBox.Items.Add(button1);

var button2 = new Button() { Content = "def" };
button2.Click += (s, e) => Console.WriteLine("button2.Click");

listBox.Items.Add(button2);

ContentPanel.Children.Add(listBox);

改为:

ContentPanel.Children.Add(
    new ListBox()
        ..Width = 200
        ..MouseEnter += ((s, e) => Console.WriteLine("MouseEnter"))
        ..Items.Add(
            new Button()
                ..Content = "abc";
                ..Click += ((s, e) => Console.WriteLine("button 1 Click")))
        ..Items.Add(
            new Button()
                ..Content = "def";
                ..Click += (s, e) => (Console.WriteLine("button 2 Click"))));

我的问题是,有没有办法模拟或近似接近C#中的方法级联?

这是我提出的一种方法。鉴于此扩展方法:

public static T Call<T>(this T obj, Action<T> proc)
{
    proc(obj);

    return obj;
}

上面的例子可以写成如下:

ContentPanel.Children.Add(
    new ListBox().Call(o => {
            o.Width = 200;
            o.MouseEnter += (s, e) => Console.WriteLine("MouseEnter");
            o.Items.Add(
                new Button().Call(b => {
                        b.Content = "abc";
                        b.Click += (s, e) => Console.WriteLine("button 1 Click"); }));
            o.Items.Add(
                new Button().Call(b => {
                    b.Content = "def";
                    b.Click += (s, e) => Console.WriteLine("button 2 Click"); })); }));

我不认为这很漂亮。 :-)但它确实能够应用流畅的风格。

2 个答案:

答案 0 :(得分:3)

我认为您可以通过使用流畅的界面达到您想要达到的目标。它允许您链接方法以在一个语句中创建和初始化对象。

你可以得到类似的东西:

Fluent fluent = new Fluent();
var panel = fluent.CreateControlPanel().Children()
                .AddListBox().SetWidth(200).AddMouseEnterEvent((s, e) => { }).Create()
                .AddTextBox().SetText("Foo").Create()
                .GetControlPanel();

这个想法是一个方法返回一个允许初始化另一个对象的对象。初始化程序链可以在任何项目上调用“终结器”方法(在Create之上),该方法返回原始对象(在Children之上)以继续添加其他对象或配置初始对象。

例如,在AddListBox中,返回ListBoxSetup类型的对象,该对象包含许多方法,如SetWidthAddMouseEnterEvent。在这种情况下,Children也将是一个特殊对象(类似于ChildSetup类型),它有一堆方法,例如AddListBoxAddTextBox。每个方法都负责创建类型为ListBoxTextBox的对象或要创建的基础对象的设置属性。 Fluent将有一个方法可以正确设置整个对象结构。

看一下这个链接: http://blog.raffaeu.com/archive/2010/06/26/how-to-write-fluent-interface-with-c-and-lambda.aspx

这是一个基础代码创建的例子,最终得到了上述内容。当然,代码可以在其架构中得到极大改进,但这只是为了示例。

public class Fluent
{
    public ControlPanelCreator CreateControlPanel()
    {
        return new ControlPanelCreator(new StackPanel(), this);
    }
}

public class ControlPanelCreator
{
    #region Fields
    private Fluent fluent;
    private Panel panel;
    #endregion

    #region Constructors
    internal ControlPanelCreator(Panel panel, Fluent fluent)
    {
        this.fluent = fluent;
        this.panel = panel;
    }
    #endregion

    #region Methods
    public ControlPanelChildrenCreator Children()
    {
        return new ControlPanelChildrenCreator(this.panel, this);
    }
    #endregion
}

public class ControlPanelChildrenCreator
{
    #region Fields
    private ControlPanelCreator panelCreator;
    private Panel panel;
    #endregion

    #region Constructors
    internal ControlPanelChildrenCreator(Panel panel, ControlPanelCreator panelCreator)
    {
        this.panel = panel;
        this.panelCreator = panelCreator;
    }
    #endregion

    #region Methods
    public ListBoxCreator AddListBox()
    {
        ListBox listBox = new ListBox();
        this.panel.Children.Add(listBox);
        return new ListBoxCreator(listBox, this);
    }

    public TextBoxCreator AddTextBox()
    {
        TextBox textBox = new TextBox();
        this.panel.Children.Add(textBox);
        return new TextBoxCreator(textBox, this);
    }

    public Panel GetControlPanel()
    {
        return this.panel;
    }
    #endregion
}

public class ListBoxCreator
{
    #region Fields
    private ListBox listbox;
    private ControlPanelChildrenCreator parentCreator;
    #endregion

    #region Constructors
    internal ListBoxCreator(ListBox listBox, ControlPanelChildrenCreator parentCreator)
    {
        this.listbox = listBox;
        this.parentCreator = parentCreator;
    }
    #endregion

    #region Methods
    public ListBoxCreator SetWidth(int width)
    {
        this.listbox.Width = width;
        return this;
    }

    public ListBoxCreator AddMouseEnterEvent(Action<object, MouseEventArgs> action)
    {
        this.listbox.MouseEnter += new MouseEventHandler(action);
        return this;
    }

    public ControlPanelChildrenCreator Create()
    {
        return this.parentCreator;
    }
    #endregion
}

public class TextBoxCreator
{
    #region Fields
    private TextBox textBox;
    private ControlPanelChildrenCreator parentCreator;
    #endregion

    #region Constructors
    internal TextBoxCreator(TextBox textBox, ControlPanelChildrenCreator parentCreator)
    {
        this.textBox = textBox;
        this.parentCreator = parentCreator;
    }
    #endregion

    #region Methods
    public TextBoxCreator SetText(string defaultText)
    {
        this.textBox.Text = defaultText;
        return this;
    }

    public ControlPanelChildrenCreator Create()
    {
        return this.parentCreator;
    }
    #endregion
}

答案 1 :(得分:0)

我支持之前的回答。我想添加的东西很少,因为我创造了类似的东西。

有两件事,要么你是单班做事。表示Form,有addcolor,addData等可能是表单有按钮而且按钮有颜色

现在在这种情况下你需要使用接口链接,意味着方法返回类型将是接口,并且所有接口都由该类实现,并且该方法仅返回“this”。

当您创建界面对象时,这将成功。然后链接它。这里很难给出示例,但如果您仍然希望我能提供示例。

如果需要进一步的详细信息,请告诉我