C#是否具有扩展属性?
例如,是否可以将扩展属性添加到将返回的DateTimeFormatInfocalled 属性?ShortDateLongTimeFormat``ShortDatePattern + " " + LongTimePattern
DateTimeFormatInfo
ShortDateLongTimeFormat``ShortDatePattern + " " + LongTimePattern
到目前为止,扩展属性的价值还不足以包含在C#标准的早期版本中。 C#7 和 C#8.0 将此视为提案的冠军,但它尚未发布,主要是因为即使已经有实现,他们也希望从一开始就做到这一点。
C#7工作列表中 有一个 扩展成员 项,因此在不久的将来可能会支持它。扩展属性的当前状态可以在Github上的相关项目下找到。
但是,还有一个更有希望的话题,那就是“扩展所有内容”,重点放在属性和静态类甚至字段上。
如本文所述,您可以使用此TypeDescriptor功能在运行时将属性附加到对象实例。但是,它没有使用标准属性的语法。 它与语法糖有点不同,它增加了定义扩展属性的可能性,例如 string Data(this MyClass instance)作为扩展方法的别名, string GetData(this MyClass instance)因为它可以将数据存储到类中。
TypeDescriptor
string Data(this MyClass instance)
string GetData(this MyClass instance)
我希望C#7将为所有功能(属性和字段)提供全功能的扩展,但是在那一点上,只有时间能证明一切。
并随时为社区的未来贡献力量。
更新:2016年8月
当dotnet团队发布了C#7.0的新功能以及Mads Torgensen的评论时:
扩展属性:我们有一个(出色的!)实习生在整个夏天作为实验与其他类型的扩展成员一起实现。我们仍然对此感兴趣,但这是一个巨大的变化,我们需要对它的价值感到自信。
似乎扩展属性和其他成员仍然是将来发布的Roslyn(包括7.0版本)中不错的选择。
更新:2017年5月
扩展成员 已关闭,因为 扩展的所有内容 均已复制,也已关闭。实际上,主要的讨论是关于广义的类型可扩展性。该功能现在 作为建议在此处 进行跟踪 , 并且已从 7.0里程碑中 删除。
更新:2017年8月-C#8.0建议的功能
尽管它仍然只是一个 建议的 功能,但是我们现在对它的语法有了更清晰的了解。请记住,这也是扩展方法的新语法:
public interface IEmployee { public decimal Salary { get; set; } } public class Employee { public decimal Salary { get; set; } } public extension MyPersonExtension extends Person : IEmployee { private static readonly ConditionalWeakTable<Person, Employee> _employees = new ConditionalWeakTable<Person, Employee>(); public decimal Salary { get { // `this` is the instance of Person return _employees.GetOrCreate(this).Salary; } set { Employee employee = null; if (!_employees.TryGetValue(this, out employee) { employee = _employees.GetOrCreate(this); } employee.Salary = value; } } } IEmployee person = new Person(); var salary = person.Salary;
与部分类相似,但是在不同的程序集中作为单独的类/类型进行编译。请注意,您还可以通过这种方式添加静态成员和运算符。如Mads Torgensen播客中所述, 该扩展将没有任何状态(因此它无法将私有实例成员添加到该类),这意味着您将无法添加链接到该实例的私有实例数据 。这样做的原因是它意味着要管理内部词典,并且可能很困难(内存管理等)。为此,您仍然可以使用前面介绍的TypeDescriptor/ ConditionalWeakTable技术,并通过属性扩展将其隐藏在一个不错的属性下。
ConditionalWeakTable
语法仍然可以更改,这暗示了这个问题。例如,extends可以替换为,for从而使某些人感觉更自然,而与Java的联系较少。
extends
for
2018年12月更新-角色,扩展和静态接口成员
扩展 并不能将其 扩展 到C#8.0,因为在GitHub票证末尾解释了一些缺点。因此,进行了改进设计的探索。在这里,Mads Torgensen解释了什么是 角色和扩展 以及它们之间的区别:
角色允许在给定类型的特定值上实现接口。扩展允许接口在代码的特定区域内在给定类型的所有值上实现。
可以在将先前的提案分为两个用例时看到。 扩展 的 新语法 如下所示:
public extension ULongEnumerable of ulong { public IEnumerator<byte> GetEnumerator() { for (int i = sizeof(ulong); i > 0; i--) { yield return unchecked((byte)(this >> (i-1)*8)); } } }
那么您将可以执行以下操作:
foreach (byte b in 0x_3A_9E_F1_C5_DA_F7_30_16ul) { WriteLine($"{e.Current:X}"); }
对于 静态接口 :
public interface IMonoid<T> where T : IMonoid<T> { static T operator +(T t1, T t2); static T Zero { get; } }
在上添加 扩展属性 ,int并将int视为IMonoid<int>:
int
IMonoid<int>
public extension IntMonoid of int : IMonoid<int> { public static int Zero => 0; }