有没有办法将类中的枚举属性设置为所有可用的枚举?

Posted

技术标签:

【中文标题】有没有办法将类中的枚举属性设置为所有可用的枚举?【英文标题】:Is there a way to set an Enumerated Property in a class to All available enums? 【发布时间】:2014-09-11 08:25:34 【问题描述】:

首先,我不知道该问题的标题是什么——我什至不知道如何陈述它。

现在回答问题。让我们使用 System.IO.FileSystemWatcher 类,您将其设置为 NotifyFilter 属性:

            this.FileSystemWatcher1.NotifyFilter = NotifyFilters.Attributes | NotifyFilters.CreationTime | NotifyFilters.FileName 
            | NotifyFilters.LastAccess | NotifyFilters.LastWrite | NotifyFilters.Security 
            | NotifyFilters.Size;

设置单个属性的代码相当多。检查NotifyFilter,它是一个枚举。是否有一种“lazy”或“shortcut”方式来一次设置所有这些属性?我知道这不是必需的,但我的好奇心被激起了。

this.FileSystemWatcher1.NotifyFilter = <NotifyFilters.All>?

【问题讨论】:

我更喜欢明确定义这种东西,例如在一个名为 All 的字段中。然后,您可以使用 ~ 排除特定的枚举值。 @GregRos 你能以某种方式证明你的意思吗? 【参考方案1】:

编写一个通用实用程序来屏蔽 c# 枚举中的所有值比人们想象的要困难得多,因为

    枚举的底层类型可以是byte、short、int或long;签名或未签名。 无法将对象直接转换为通用枚举,可能是因为 C# 中没有内置枚举约束。必须先装箱,然后再拆箱。 所有枚举实用程序都来自 c# 1.0,因此相当粗糙。

这是我能做的最好的。我使用了以下内容:

    Cast Int to Generic Enum in C# Enum type constraints in C#

这在 c++/CLI 中会更容易吗?

/// <summary>
/// Contains generic utilities for enums, constrained for enums only.
/// </summary>
public sealed class EnumHelper : Enums<Enum>

    private EnumHelper()
    
    


/// <summary>
/// For use by EnumHelper, not for direct use.
/// </summary>
public abstract class Enums<TEnumBase> where TEnumBase : class, IConvertible

    private static void ThrowOnEnumWithoutFlags<TEnum>() where TEnum : struct, TEnumBase
    
        var attributes = typeof(TEnum).GetCustomAttributes(typeof(FlagsAttribute), false);
        if (attributes.Length == 0)
        
            throw (new ArgumentException("The generic argument [<T>] must be an enumeration with the [FlagsAttribute] applied.", "T: " + typeof(TEnum).FullName));
        
    

    public static TEnum GetAll<TEnum>() where TEnum : struct, TEnumBase
    
        ThrowOnEnumWithoutFlags<TEnum>();
        var underlyingType = Enum.GetUnderlyingType(typeof(TEnum));
        if (underlyingType == typeof(ulong))
        
            ulong value = 0;
            foreach (var v in Enum.GetValues(typeof(TEnum)))
                // Not sure I need the culture but Microsoft passes it in Enum.ToUInt64(Object value) - http://referencesource.microsoft.com/#mscorlib/system/enum.cs
                value |= Convert.ToUInt64(v, CultureInfo.InvariantCulture); 
            return (TEnum)Enum.ToObject(typeof(TEnum), value);
        
        else
        
            long value = 0;
            foreach (var v in Enum.GetValues(typeof(TEnum)))
                // Not sure I need the culture but Microsoft passes it in Enum.ToUInt64(Object value) - http://referencesource.microsoft.com/#mscorlib/system/enum.cs
                value |= Convert.ToInt64(v, CultureInfo.InvariantCulture);
            return (TEnum)Enum.ToObject(typeof(TEnum), value);
        
    

我已经测试过 byte、sbyte、short、int、long、ushort、uint 和 ulong 类型。

更新按照 hvd 的建议进行了简化。

【讨论】:

+1 您确实需要 longulong 的单独处理程序来处理超出 long 范围的值,但对于所有其他类型,您可以使用 long版本:只需使用Convert.ToInt64Enum.ToObject 来执行转换。 (为了保持一致性,您可以使用Convert.ToUInt64Enum.ToObject 来表示ulong 版本。)【参考方案2】:

你总是可以做这样的事情,

NotifyFilter ret = 0;
foreach(NotifyFilter v in Enum.GetValues(typeof(NotifyFilter)))

    ret |= v;   

不幸的是,我不知道更好的方法。但是你总是可以把它放在一个通用的实用方法中。

private static T GetAll<T>() where T : struct, IConvertible

    if (!typeof(T).IsEnum)
    
       throw new NotSupportedException(); // You'd want something better here, of course.
    

    long ret = 0; // you could determine the type with reflection, but it might be easier just to use multiple methods, depending on how often you tend to use it.
    foreach(long v in Enum.GetValues(typeof(T)))
    
        ret |= v;
    

    return (T)ret;

【讨论】:

仅供参考,您可以说foreach(NotifyFilter in Enum.GetValues(typeof(NotifyFilter))) 将演员表保存在 foreach 体内。 好一个。我正在处理一些主要的个人项目,并希望保持我的应用程序快速。我没有想到使用通用方法来解决这个问题。 T 限制为where T: struct, IConvertible 并检查if (!typeof(T).IsEnum) throw ... (T)ret 无法为我编译——但我​​坚持使用 c# 3.0。它适用于 4.0/4.5 吗? @Noobgrammer 也许可以设置所有标志,即使是那些未使用或未命名的标志。在这种情况下,您只需执行this.FileSystemWatcher1.NotifyFilter = (NotifyFilters)(~0); 或(等效)this.FileSystemWatcher1.NotifyFilter = (NotifyFilters)(-1);。但一定要检查NotifyFilters的底层类型是否为int(it is)。【参考方案3】:

如果枚举应用了Flags 属性,您可以添加与所需枚举成员等效的数字,然后分配它。

在你问题的例子中,总结 4+64+32+16+8 = 124 然后你可以写

this.FileSystemWatcher1.NotifyFilter = (NotifyFilters) 124;

当然,这会降低枚举的可读性并严重导致维护问题,但正如您所说,这是为我们的懒人准备的:

public class Program

    [Flags]
    enum Enum:int
    
        a = 1,
        b = 2
    

    static void Main(string[] args)
    
        Enum x = (Enum) 3;
        Console.WriteLine(x.ToString());

    

输出:

a,b

更新:

我差点忘了。您当然可以将此字符串传递给Enum.Parse 并获得所需的值:

this.FileSystemWatcher1.NotifyFilter = (NotifyFilter) Enum.Parse(typeof(NotifyFilter), 
"Attributes,CreationTime,FileName,LastAccess,LastWrite,Security,Size"); 

【讨论】:

这很奇怪。不知道你能做到这一点 它仍然把我吓坏了,但正如你所说:好奇心,当然还有懒惰。 我可以想象这个在意大利面条代码中的着陆,以及下一个开发人员从 27 层楼的建筑中跳下 没错。我自己讨厌这个,但在一些紧张的地方会派上用场。【参考方案4】:

如果不编写自己的方法,就没有办法——正如其他地方正确回答的那样。

如果枚举是你要更改的,你可以添加一个新值

All = ~0

【讨论】:

以上是关于有没有办法将类中的枚举属性设置为所有可用的枚举?的主要内容,如果未能解决你的问题,请参考以下文章

有没有办法使用 Template Haskell 枚举模块中的所有函数?

枚举嵌入资源目录中的文件

java 枚举

java中的枚举到底有啥作用?

在“方法/事件/枚举下拉菜单”中禁用分类

C++ 中的全局枚举