有谁知道如何将枚举值转换为人类可读的值?
例如:
ThisIsValueA应为“This is Value A”。
答案 0 :(得分:17)
将某个Ian Horwill留在blog post long ago的某个vb代码段转换为...我已经成功地在生产中使用了它。
/// <summary>
/// Add spaces to separate the capitalized words in the string,
/// i.e. insert a space before each uppercase letter that is
/// either preceded by a lowercase letter or followed by a
/// lowercase letter (but not for the first char in string).
/// This keeps groups of uppercase letters - e.g. acronyms - together.
/// </summary>
/// <param name="pascalCaseString">A string in PascalCase</param>
/// <returns></returns>
public static string Wordify(string pascalCaseString)
{
Regex r = new Regex("(?<=[a-z])(?<x>[A-Z])|(?<=.)(?<x>[A-Z])(?=[a-z])");
return r.Replace(pascalCaseString, " ${x}");
}
(要求'使用System.Text.RegularExpressions;')
因此:
Console.WriteLine(Wordify(ThisIsValueA.ToString()));
会回来,
"This Is Value A".
它比提供描述属性更简单,更省事。
只有当你需要提供一个间接层(问题没有要求)时,属性才有用。
答案 1 :(得分:10)
Enums上的.ToString在C#中相对较慢,与GetType()相比。名称(甚至可能在封面下使用)。
如果您的解决方案需要非常快速或高效,您可能最好将您的转换缓存在静态字典中,并从那里查找它们。
@ Leon的代码的小改编,以利用C#3。这作为枚举的扩展是有意义的 - 如果你不想让所有这些都混乱,你可以将它限制为特定的类型。
public static string Wordify(this Enum input)
{
Regex r = new Regex("(?<=[a-z])(?<x>[A-Z])|(?<=.)(?<x>[A-Z])(?=[a-z])");
return r.Replace( input.ToString() , " ${x}");
}
//then your calling syntax is down to:
MyEnum.ThisIsA.Wordify();
答案 2 :(得分:4)
我见过的大多数这样的例子都涉及用[描述]属性标记你的枚举值,并使用反射在值和描述之间进行“转换”。这是一篇关于它的老博文:
http://geekswithblogs.net/rakker/archive/2006/05/19/78952.aspx
答案 3 :(得分:4)
您可以从System.Reflection的“Attribute”类继承来创建自己的“Description”类。像这样(来自here):
using System;
using System.Reflection;
namespace FunWithEnum
{
enum Coolness : byte
{
[Description("Not so cool")]
NotSoCool = 5,
Cool, // since description same as ToString no attr are used
[Description("Very cool")]
VeryCool = NotSoCool + 7,
[Description("Super cool")]
SuperCool
}
class Description : Attribute
{
public string Text;
public Description(string text)
{
Text = text;
}
}
class Program
{
static string GetDescription(Enum en)
{
Type type = en.GetType();
MemberInfo[] memInfo = type.GetMember(en.ToString());
if (memInfo != null && memInfo.Length > 0)
{
object[] attrs = memInfo[0].GetCustomAttributes(typeof(Description), false);
if (attrs != null && attrs.Length > 0)
return ((Description)attrs[0]).Text;
}
return en.ToString();
}
static void Main(string[] args)
{
Coolness coolType1 = Coolness.Cool;
Coolness coolType2 = Coolness.NotSoCool;
Console.WriteLine(GetDescription(coolType1));
Console.WriteLine(GetDescription(coolType2));
}
}
}
答案 4 :(得分:2)
您还可以查看一下这篇文章:http://www.codeproject.com/KB/cs/enumdatabinding.aspx
它专门针对数据绑定,但展示了如何使用属性来修饰枚举值,并提供了一个“GetDescription”方法来检索属性的文本。使用内置描述属性的问题在于该属性还有其他用途/用户,因此描述可能会出现在您不希望的位置。自定义属性解决了这个问题。
答案 5 :(得分:0)
我发现最好用低分来定义你的枚举值,这样ThisIsValueA就是This_Is_Value_A然后你可以做一个enumValue.toString()。替换(“_”,“”)其中enumValue是你的变量。
答案 6 :(得分:0)
向每个枚举添加Description
属性的另一种方法是创建扩展方法。重用亚当的“ Coolness”枚举:
public enum Coolness
{
NotSoCool,
Cool,
VeryCool,
SuperCool
}
public static class CoolnessExtensions
{
public static string ToString(this Coolness coolness)
{
switch (coolness)
{
case Coolness.NotSoCool:
return "Not so cool";
case Coolness.Cool:
return "Cool";
case Coolness.VeryCool:
return "Very cool";
case Coolness.SuperCool:
return Properties.Settings.Default["SuperCoolDescription"].ToString();
default:
throw new ArgumentException("Unknown amount of coolness", nameof(coolness));
}
}
}
尽管这意味着描述与实际值相距甚远,但是它允许您使用本地化为每种语言打印不同的字符串,例如在我的VeryCool
示例中。