c# - 有什么方法可以在 C# 的对象初始化程序 block 中使用扩展方法

标签 c# linq extension-methods initializer

下面的简单演示捕获了我正在尝试做的事情。在实际程序中,我必须使用对象初始化程序 block ,因为它正在读取 LINQ to SQL 选择表达式中的列表,并且有一个值我想从数据库中读取并存储在对象上,但是对象没有我可以为该值设置的简单属性。相反,它有一个 XML 数据存储。

看起来我无法在对象初始化程序 block 中调用扩展方法,而且我无法使用扩展方法附加属性。

那么我对这种方法不走运吗?唯一的选择似乎是说服基类的所有者针对这种情况修改它。

我有一个现有的解决方案,我将 BaseDataObject 子类化,但这也有一些问题没有出现在这个简单的例子中。对象被保留并恢复为 BaseDataObject - 转换和测试会变得复杂。

public class BaseDataObject
{

    // internal data store
    private Dictionary<string, object> attachedData = new Dictionary<string, object>();

    public void SetData(string key, object value)
    {
        attachedData[key] = value;
    }

    public object GetData(string key)
    {
        return attachedData[key];
    }

    public int SomeValue { get; set; }
    public int SomeOtherValue { get; set; }

}

public static class Extensions
{
    public static void SetBarValue(this BaseDataObject dataObject,
                                        int            barValue)
    {
        /// Cannot attach a property to BaseDataObject?
        dataObject.SetData("bar", barValue);
    }
}

public class TestDemo
{

    public void CreateTest()
    {
        // this works
        BaseDataObject test1 = new BaseDataObject 
        { SomeValue = 3, SomeOtherValue = 4 };

        // this does not work - it does not compile
        // cannot use extension method in the initialiser block
        // cannot make an exension property  
        BaseDataObject test2 = new BaseDataObject { SomeValue = 3, SomeOtherValue = 4, SetBarValue(5) };
    }
}

其中一个答案(来自 mattlant)建议使用流畅的界面样式扩展方法。例如:

// fluent interface style
public static BaseDataObject SetBarValueWithReturn(this BaseDataObject dataObject, int barValue)
{
    dataObject.SetData("bar", barValue);
    return dataObject;
}

// this works
BaseDataObject test3 = (new BaseDataObject { SomeValue = 3, SomeOtherValue = 4 }).SetBarValueWithReturn(5);

但这会在 LINQ 查询中起作用吗?

最佳答案

对象初始化器只是语法糖,需要一个聪明的编译器,并且在当前实现中,您不能调用初始化器中的方法。

var x = new BaseDataObject { SomeValue = 3, SomeOtherValue = 4 };

会让编译器变成这样:

BaseDataObject tempObject = new BaseDataObject();
tempObject.SomeValue = 3;
tempObject.SomeOtherValue = 4;
BaseDataObject x = tempObject;

不同之处在于不会有任何同步问题。变量 x get 立即分配了完全分配的 BaseDataObject,您不能在对象初始化期间弄乱它。

您可以在创建对象后调用扩展方法:

var x = new BaseDataObject { SomeValue = 3, SomeOtherValue = 4 };
x.SetBarValue()

您可以将 SetBarValue 更改为可以在初始化期间分配的具有 get/set 的属性:

public int BarValue
{
    set
    {
        //Value should be ignored
    }
}

或者,您可以子类化/使用外观模式将方法添加到您的对象上:

public class DataObjectWithBarValue : BaseDataObject
{
    public void BarValue
    {
        set
        {
            SetData("bar", value);
        }
        get
        {
            (int) GetData("bar");
        }
    }
}

关于c# - 有什么方法可以在 C# 的对象初始化程序 block 中使用扩展方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/132245/

相关文章:

c# - Netstandard 无法从 nuget System.Net.Http 转换为 GAC 版本

c# - 在 WPF 中运行我的应用程序时如何禁用 "PRINT SCREEN"按钮?

c# - 网络摄像头时间不准确,设置曝光时间

c# - 列出与另一个列表条件相关的操作

c# - 使用 OrderBy 的 EF 查询失败

c# - 控制台应用程序文本始终在最前面

C# LINQ 比较列表值

c# - FindAll 与 Where 扩展方法

c# - 为什么必须在静态类中定义 C# 扩展方法?

groovy - Groovy是否具有C#中的扩展方法?