RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 572077
Accepted
XelaNimed
XelaNimed
Asked:2020-09-30 16:35:06 +0000 UTC2020-09-30 16:35:06 +0000 UTC 2020-09-30 16:35:06 +0000 UTC

通过文本值从枚举中提取常量

  • 772

有以下方法,它enum通过其字符串表示形式返回一个命名常量,或者在失败时返回一个默认值:

static TEnum GetEnum<TEnum>(string key, TEnum defaultValue = default(TEnum), bool ignoreCase = true) 
where TEnum : struct, IConvertible {
    TEnum result;
    return Enum.TryParse<TEnum>(key, ignoreCase, out result) ? result : defaultValue;
}

问题:为什么下面的代码会为一个不存在的常量返回错误的结果?如何解决?

enum Nums { none = 0, one, two, three, four, five }

Nums res;

res = GetEnum<Nums>("one");
Console.WriteLine($"{res} : {res == Nums.one}");   // one : True
res = GetEnum<Nums>("six");
Console.WriteLine($"{res} : {res == Nums.none}");  // none : True
res = GetEnum<Nums>("123");
Console.WriteLine($"{res} : {res == Nums.none}");  // 123 : False ???

我假设这是由于 string 的数字解释而发生的123,但仍然不清楚为什么会发生这种情况以及如何解决它?

c#
  • 3 3 个回答
  • 10 Views

3 个回答

  • Voted
  1. Best Answer
    VladD
    2020-09-30T17:07:52Z2020-09-30T17:07:52Z

    这种奇怪的行为被记录在案:-\

    如果value是一个不对应于 的命名常量的名称TEnum,则该方法返回false。如果value是不表示TEnum枚举的基础值的整数的字符串表示形式,则该方法返回一个枚举成员,其基础值是转换为整数类型的值。如果不需要此行为,请调用该IsDefined方法以确保整数的特定字符串表示实际上是 的成员TEnum。

    看起来您需要像这样重写函数:

    static TEnum GetEnum<TEnum>(
        string key,
        TEnum defaultValue = default(TEnum),
        bool ignoreCase = true)
            where TEnum : struct, IConvertible
    {
        TEnum result;
        var b = Enum.TryParse<TEnum>(key, ignoreCase, out result) &&
                  Enum.IsDefined(typeof(TEnum), result);
        return b ? result : defaultValue;
    }
    

    更新:修改后的函数未通过字符串"1"和的测试"one,one"。所以看起来需要更多检查:

    static TEnum GetEnum<TEnum>(
        string key,
        TEnum defaultValue = default(TEnum),
        bool ignoreCase = true)
            where TEnum : struct, IConvertible
    {
        TEnum result;
        var b =
            Enum.TryParse<TEnum>(key, ignoreCase, out result) &&
            Enum.IsDefined(typeof(TEnum), result) &&
            string.Equals(result.ToString(), key, StringComparison.OrdinalIgnoreCase);
        return b ? result : defaultValue;
    }
    

    更新:上面的代码没有处理GetEnum<System.Windows.Forms.MessageBoxIcon>("Error"),因为里面的常量System.Windows.Forms.MessageBoxIcon是重复的!作为解决方案,我们可以查询所有名称的列表并进行比较。最终版本:

    static TEnum GetEnum<TEnum>(
        string key,
        TEnum defaultValue = default(TEnum),
        bool ignoreCase = true)
            where TEnum : struct, IConvertible
    {
        TEnum result;
        if (Enum.GetNames(typeof(TEnum))
                .Contains(key, ignoreCase ? StringComparer.OrdinalIgnoreCase :
                                            StringComparer.Ordinal) &&
            Enum.TryParse<TEnum>(key, ignoreCase, out result))
            return result;
        else       
            return defaultValue;
    }
    

    [你可以通过支票逃脱Enum.IsDefined(typeof(TEnum), key)(不是result!),但不可能要求忽略其中的大小写。]


    更新:作为评论中讨论的结果,您可以使用不那么漂亮但更高效的代码:

    static TEnum GetEnum<TEnum>(
        string key,
        TEnum defaultValue = default(TEnum),
        bool ignoreCase = true)
            where TEnum : struct, IConvertible
    {
        var comparisonType = ignoreCase ? StringComparison.OrdinalIgnoreCase :
                                          StringComparison.Ordinal;
        var names = Enum.GetNames(typeof(TEnum));
        foreach (var name in names)
        {
            if (name.Equals(key, comparisonType))
                return (TEnum)Enum.Parse(typeof(TEnum), key, ignoreCase);
        }
        return defaultValue;
    }
    

    它的本质是一样的,只是一些操作被“画”出来了。评论中的基准。


    如果您需要极端的微优化并了解缓存问题,这是 @Pavel Mayorov 提出的一个更快的缓存选项:

    private class Holder<TEnum>
    {
        // относительный порядок элементов в массивах гарантирован документацией
        public static readonly TEnum[] values = (TEnum[])Enum.GetValues(typeof(TEnum));
        public static readonly string[] names = Enum.GetNames(typeof(TEnum));
    }
    
    static TEnum GetEnum<TEnum>(
        string key,
        TEnum defaultValue = default(TEnum),
        bool ignoreCase = true)
            where TEnum : struct, IConvertible
    {
        var comparisonType = ignoreCase ? StringComparison.OrdinalIgnoreCase :
                                          StringComparison.Ordinal;
        for (int i = 0; i < Holder<TEnum>.names.Length; i++)
        {
            if (Holder<TEnum>.names[i].Equals(key, comparisonType))
                return Holder<TEnum>.values[i];
        }
        return defaultValue;
    }
    
    • 9
  2. andreycha
    2020-09-30T17:15:01Z2020-09-30T17:15:01Z

    这是因为里面 TryParse有两个分支:

    • 如果传递的字符串确实是一个字符串值(不是数字),那么该方法会诚实地将它与 enam 中定义的所有成员名称进行比较。
    • 如果传递的字符串结果是一个数字,则该方法返回 enam 值,该值与指定的数字相匹配。但是,如果 enam 中没有具有该值的成员,则返回数字本身。

    因此,如果您需要从字符串中解析 enam,而该字符串可以同时包含 enam 成员的名称及其值,那么您需要另外使用IsDefined().

    • 5
  3. Slavyan32
    2020-09-30T16:58:17Z2020-09-30T16:58:17Z

    这是由于数字 123 不在集合中,并且由于 123 不等于 0,因此结果为 False。尝试指定 0 而不是 123,那么结果将为 True。

    • -1

相关问题

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    如何停止编写糟糕的代码?

    • 3 个回答
  • Marko Smith

    onCreateView 方法重构

    • 1 个回答
  • Marko Smith

    通用还是非通用

    • 2 个回答
  • Marko Smith

    如何访问 jQuery 中的列

    • 1 个回答
  • Marko Smith

    *.tga 文件的组重命名(3620 个)

    • 1 个回答
  • Marko Smith

    内存分配列表C#

    • 1 个回答
  • Marko Smith

    常规赛适度贪婪

    • 1 个回答
  • Marko Smith

    如何制作自己的自动完成/自动更正?

    • 1 个回答
  • Marko Smith

    选择斐波那契数列

    • 2 个回答
  • Marko Smith

    所有 API 版本中的通用权限代码

    • 2 个回答
  • Martin Hope
    jfs *(星号)和 ** 双星号在 Python 中是什么意思? 2020-11-23 05:07:40 +0000 UTC
  • Martin Hope
    hwak 哪个孩子调用了父母的静态方法?还是不可能完成的任务? 2020-11-18 16:30:55 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    user207618 Codegolf——组合选择算法的实现 2020-10-23 18:46:29 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    Arch ArrayList 与 LinkedList 的区别? 2020-09-20 02:42:49 +0000 UTC
  • Martin Hope
    iluxa1810 哪个更正确使用:if () 或 try-catch? 2020-08-23 18:56:13 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5