Как вы получаете максимальное значение перечисления?
Как вы получаете максимальное значение перечисления?
Ответы:
Enum.GetValues (), кажется, возвращает значения по порядку, так что вы можете сделать что-то вроде этого:
// given this enum:
public enum Foo
{
Fizz = 3,
Bar = 1,
Bang = 2
}
// this gets Fizz
var lastFoo = Enum.GetValues(typeof(Foo)).Cast<Foo>().Last();
редактировать
Для тех, кто не хочет читать комментарии: Вы также можете сделать это следующим образом:
var lastFoo = Enum.GetValues(typeof(Foo)).Cast<Foo>().Max();
... который будет работать, когда некоторые из ваших значений enum отрицательны.
Я согласен с ответом Мэтта. Если вам нужны только значения min и max int, то вы можете сделать это следующим образом.
Максимум:
Enum.GetValues(typeof(Foo)).Cast<int>().Max();
Минимум:
Enum.GetValues(typeof(Foo)).Cast<int>().Min();
Согласно ответу Мэтта Гамильтона, я подумал о создании для него метода Extension.
Так ValueType
не принят в качестве общего ограничения параметров типа, я не нашел лучший способ ограничить T
до , Enum
но ниже.
Любые идеи будут по достоинству оценены.
PS. пожалуйста, игнорируйте мою неявность VB, я люблю использовать VB таким образом, это сила VB, и поэтому я люблю VB.
Хоева, вот оно:
static void Main(string[] args)
{
MyEnum x = GetMaxValue<MyEnum>(); //In newer versions of C# (7.3+)
MyEnum y = GetMaxValueOld<MyEnum>();
}
public static TEnum GetMaxValue<TEnum>()
where TEnum : Enum
{
return Enum.GetValues(typeof(TEnum)).Cast<TEnum>().Max();
}
//When C# version is smaller than 7.3, use this:
public static TEnum GetMaxValueOld<TEnum>()
where TEnum : IComparable, IConvertible, IFormattable
{
Type type = typeof(TEnum);
if (!type.IsSubclassOf(typeof(Enum)))
throw new
InvalidCastException
("Cannot cast '" + type.FullName + "' to System.Enum.");
return (TEnum)Enum.ToObject(type, Enum.GetValues(type).Cast<int>().Last());
}
enum MyEnum
{
ValueOne,
ValueTwo
}
Public Function GetMaxValue _
(Of TEnum As {IComparable, IConvertible, IFormattable})() As TEnum
Dim type = GetType(TEnum)
If Not type.IsSubclassOf(GetType([Enum])) Then _
Throw New InvalidCastException _
("Cannot cast '" & type.FullName & "' to System.Enum.")
Return [Enum].ToObject(type, [Enum].GetValues(type) _
.Cast(Of Integer).Last)
End Function
Это немного nitpicky , но фактическое максимальное значение любого enum
является Int32.MaxValue
(при условии , что это enum
происходит от int
). Совершенно законно приводить любое Int32
значение к любому enum
независимо от того, действительно ли оно объявило члена с этим значением.
Допустимо:
enum SomeEnum
{
Fizz = 42
}
public static void SomeFunc()
{
SomeEnum e = (SomeEnum)5;
}
После того, как попробовал в другой раз, я получил этот метод расширения:
public static class EnumExtension
{
public static int Max(this Enum enumType)
{
return Enum.GetValues(enumType.GetType()).Cast<int>().Max();
}
}
class Program
{
enum enum1 { one, two, second, third };
enum enum2 { s1 = 10, s2 = 8, s3, s4 };
enum enum3 { f1 = -1, f2 = 3, f3 = -3, f4 };
static void Main(string[] args)
{
Console.WriteLine(enum1.one.Max());
}
}
При использовании функции Last не удалось получить максимальное значение. Использовать функцию «макс» можно. Подобно:
class Program
{
enum enum1 { one, two, second, third };
enum enum2 { s1 = 10, s2 = 8, s3, s4 };
enum enum3 { f1 = -1, f2 = 3, f3 = -3, f4 };
static void Main(string[] args)
{
TestMaxEnumValue(typeof(enum1));
TestMaxEnumValue(typeof(enum2));
TestMaxEnumValue(typeof(enum3));
}
static void TestMaxEnumValue(Type enumType)
{
Enum.GetValues(enumType).Cast<Int32>().ToList().ForEach(item =>
Console.WriteLine(item.ToString()));
int maxValue = Enum.GetValues(enumType).Cast<int>().Max();
Console.WriteLine("The max value of {0} is {1}", enumType.Name, maxValue);
}
}
По согласованию с Мэтью Дж Салливаном, для C #:
Enum.GetValues(typeof(MyEnum)).GetUpperBound(0);
Я действительно не уверен, почему кто-то хотел бы использовать:
Enum.GetValues(typeof(MyEnum)).Cast<MyEnum>().Last();
... Как дословно, семантически говоря, это не имеет особого смысла? (всегда хорошо иметь разные способы, но я не вижу выгоды в последнем.)
Существуют методы для получения информации о перечисляемых типах в System.Enum.
Итак, в проекте VB.Net в Visual Studio я могу напечатать «System.Enum». и intellisense воспитывает все виды доброты.
В частности, одним из методов является System.Enum.GetValues (), который возвращает массив перечисляемых значений. Получив массив, вы сможете делать все, что подходит для ваших конкретных обстоятельств.
В моем случае мои перечисляемые значения начинались с нуля и не пропускали никаких чисел, поэтому, чтобы получить максимальное значение для моего перечисления, мне просто нужно знать, сколько элементов было в массиве.
Фрагменты кода VB.Net:
'''''''
Enum MattType
zerothValue = 0
firstValue = 1
secondValue = 2
thirdValue = 3
End Enum
'''''''
Dim iMax As Integer
iMax = System.Enum.GetValues(GetType(MattType)).GetUpperBound(0)
MessageBox.Show(iMax.ToString, "Max MattType Enum Value")
'''''''
GetUpperBound()
извлекает максимально возможный индекс в массиве, возвращенном GetValues()
, а не наибольшее значение, хранящееся в этом массиве.
В F # с помощью вспомогательной функции для преобразования перечисления в последовательность:
type Foo =
| Fizz = 3
| Bang = 2
// Helper function to convert enum to a sequence. This is also useful for iterating.
// stackoverflow.com/questions/972307/can-you-loop-through-all-enum-values-c
let ToSeq (a : 'A when 'A : enum<'B>) =
Enum.GetValues(typeof<'A>).Cast<'B>()
// Get the max of Foo
let FooMax = ToSeq (Foo()) |> Seq.max
Запуск это ...
> Тип Foo = | Fizz = 3 | Взрыв = 2 > val ToSeq: 'A -> seq <' B> когда 'A: enum <' B> > val FooMax: Foo = Fizz
when 'A : enum<'B>
Не требуется компилятор для определения, но требуются для любого использования ToSeq, даже действительным перечислимого типа.
Его нельзя использовать при любых обстоятельствах, но я часто сам определяю максимальное значение:
enum Values {
one,
two,
tree,
End,
}
for (Values i = 0; i < Values.End; i++) {
Console.WriteLine(i);
}
var random = new Random();
Console.WriteLine(random.Next((int)Values.End));
Конечно, это не будет работать, когда вы используете пользовательские значения в перечислении, но часто это может быть простым решением.
Я использовал следующее, когда мне нужно минимальное и максимальное значения моего перечисления. Я просто установил минимальное значение, равное минимальному значению перечисления, и максимальное значение, равное максимальному значению перечисления, поскольку сами значения перечисления.
public enum ChannelMessageTypes : byte
{
Min = 0x80, // Or could be: Min = NoteOff
NoteOff = 0x80,
NoteOn = 0x90,
PolyKeyPressure = 0xA0,
ControlChange = 0xB0,
ProgramChange = 0xC0,
ChannelAfterTouch = 0xD0,
PitchBend = 0xE0,
Max = 0xE0 // Or could be: Max = PitchBend
}
// I use it like this to check if a ... is a channel message.
if(... >= ChannelMessageTypes.Min || ... <= ChannelMessages.Max)
{
Console.WriteLine("Channel message received!");
}