我有一个包含多个标志的XML,其中一些是无符号的32位整数,另一些是无符号的64位整数。其中一些是用逗号分隔的列表编写的,另一些是十六进制的。
见这个例子:
<Color>Blue,Red</Color>
<Color>0xC</Color>
由于我不想编写解析每个枚举的方法,我决定使用泛型方法。但Visual Studio不会让我构建解决方案。这是我的方法:
public static T ParseFlags<T>(string value) where T : struct
{
T result = (T)((object)0);
string[] array;
// Remove white spaces and delimit string if it is comma-separated
if (ParseDelimitedString(value, ',', out array))
{
for (int i = 0; i < array.Length; i++)
{
T flag = (T)((object)0);
// Check if value is member of enumeration
if (Enum.TryParse<T>(array[i], out flag))
{
result |= (T)((object)flag);
}
}
}
else
{
switch (Type.GetTypeCode(Enum.GetUnderlyingType(typeof(T))))
{
// Remove hex characters and parse node's inner text
case TypeCode.UInt32:
result = (T)((object)ParseUint(value));
break;
case TypeCode.UInt64:
result = (T)((object)ParseUlong(value));
break;
}
}
return result;
}
我得到的错误信息是:
错误1运算符'| ='不能应用于'T'类型的操作数 'T'
有办法做到这一点吗?
答案 0 :(得分:7)
您正在做很多可以为您完成的工作。例如,如果您使用enum
声明FlagsAttribute
,那么Enum.Parse
将为您解析逗号分隔值。
public static T ParseFlags<T>(string value) where T : struct
{
T result;
ulong temp;
if (Enum.TryParse(value, out result))
{
return result;
}
string hexNum = value.StartsWith("0x") ? value.Substring(2) : value;
if (ulong.TryParse(hexNum, NumberStyles.HexNumber, null, out temp))
{
return (T)Enum.ToObject(typeof(T), temp);
}
throw new ArgumentException("value could not be parsed");
}
我使用short
,int
和ulong
支持值的各种Flags枚举类型对此进行了测试。
答案 1 :(得分:1)
如果你知道你正在解析什么类型的枚举:
[Fact]
public void when_parsing_options_then_can_combine_flags()
{
var values = "Singleline | Compiled";
var options = values.Split(new[] { '|' }, StringSplitOptions.RemoveEmptyEntries)
.Select(value => (RegexOptions)Enum.Parse(typeof(RegexOptions), value))
.Aggregate(RegexOptions.None, (current, value) => current |= value);
Assert.Equal(RegexOptions.Singleline | RegexOptions.Compiled, options);
}
答案 2 :(得分:0)
试试这个:
public static T ParseFlags<T>(string value) where T : struct
{
long result = 0L;
string[] array;
// Remove white spaces and delimit string if it is comma-separated
if (ParseDelimitedString(value, ',', out array))
{
for (int i = 0; i < array.Length; i++)
{
T flag = default(T);
// Check if value is member of enumeration
if (Enum.TryParse<T>(array[i], out flag))
{
result |= (long)flag;
}
}
}
else
{
switch (Type.GetTypeCode(Enum.GetUnderlyingType(typeof(T))))
{
// Remove hex characters and parse node's inner text
case TypeCode.UInt32:
result = ParseUint(value);
break;
case TypeCode.UInt64:
result = ParseUlong(value);
break;
}
}
return (T)((object)result);
}
希望它有所帮助。
答案 3 :(得分:0)
“| =”该片段中的用法让我觉得你的意思是将Enum用作bitset而不仅仅是任何旧的Enum。如果这是真的,你应该做一个小改动 - 将本地“result”声明为int并适当调整转换,你的return语句应该是“return(T)(object)result;”。与“|”有关的那一行看起来像:“result | =(int)(object)flag;”。也许,有一个更好的答案但是,请注意enums are integers和你的bitset情景在这个解决方案中得到了很好的解决,除非有些情况我错过了或你没有陈述。