M
Matt
Hi,
Recently we had some code like this cause a failure:
MyEnum myEnum = (MyEnum) (int) dt[row][field];
i.e. reading an int out of the database and casting it into a
type-safe enum.
The thought behind this construct was to enforce type safety and
detect data corruption, which it only partly succeeded in doing.
The bug in our code was that that the wrong field was read from the
datatable, and this allowed an int that was not a defined enum element
to be read instead.
I had thought that this would cause an InvalidCastException, but it
appears that it does not as this test code demonstrates:
class Class1
{
enum MyEnum
{
Zero,
One,
Two,
Three
}
[STAThread]
static void Main(string[] args)
{
MyEnum myEnumA = (MyEnum) 99;
MyEnum myEnumB = (MyEnum) 1;
Console.WriteLine(myEnumA.ToString());
Console.WriteLine(myEnumB.ToString());
Console.Read();
}
}
This produces the following output:
99
One
The bug was eventually caught by a switch that throws an exception on
the default case, so our code managed to enforce safety eventually
It's not really a problem now that we know about it, but it doesn't
seem to be very intuitive as standard behaviour, given that in other
respects enums *are* type-safe, and that Enum.IsDefined can be used to
test whether enum elements are defined or not -- it would be nice to
be able to choose to have that check done automatically, as I think it
would make for safer code.
I'd be interested to hear what others think of this enum behaviour.
Regards,
Matt
Recently we had some code like this cause a failure:
MyEnum myEnum = (MyEnum) (int) dt[row][field];
i.e. reading an int out of the database and casting it into a
type-safe enum.
The thought behind this construct was to enforce type safety and
detect data corruption, which it only partly succeeded in doing.
The bug in our code was that that the wrong field was read from the
datatable, and this allowed an int that was not a defined enum element
to be read instead.
I had thought that this would cause an InvalidCastException, but it
appears that it does not as this test code demonstrates:
class Class1
{
enum MyEnum
{
Zero,
One,
Two,
Three
}
[STAThread]
static void Main(string[] args)
{
MyEnum myEnumA = (MyEnum) 99;
MyEnum myEnumB = (MyEnum) 1;
Console.WriteLine(myEnumA.ToString());
Console.WriteLine(myEnumB.ToString());
Console.Read();
}
}
This produces the following output:
99
One
The bug was eventually caught by a switch that throws an exception on
the default case, so our code managed to enforce safety eventually

It's not really a problem now that we know about it, but it doesn't
seem to be very intuitive as standard behaviour, given that in other
respects enums *are* type-safe, and that Enum.IsDefined can be used to
test whether enum elements are defined or not -- it would be nice to
be able to choose to have that check done automatically, as I think it
would make for safer code.
I'd be interested to hear what others think of this enum behaviour.
Regards,
Matt