I'm working with attributes at the moment. I often run into the error 'An attribute argument must be a constant expression, typeof expression or array creation expression of an attribute parameter type.' I don't really know what 'constant expression' means.
It would have been really useful to be able to pass in a Func<MyType, bool>
to the attribute (to be consumed by the code which executes when the attribute is present). But alas, no. I don't understand why that type cannot be placed in assembly metadata, which I assume is the reason I cannot pass it into the attribute.
Can anyone give me any ideas?
Constant expressions are values determined solely at compile-time, including string concatenation of other constant expressions, arithmetic etc.
So for example "" is a constant expression, but String.Empty
isn't.
String
is the only reference type to support a non-null constant expression. For value types, the primitive types (int
etc) and decimal
support constant expressions... although you can't use decimal
in attributes, as it's not a primitive in the CLR. (You can't even specify decimal
as a parameter type in an attribute constructor.)
See section 7.19 of the C# 4 spec for more information.
The compiler needs to be able to create the Attributes at compile time, since they are embedded in your assembly with their actual data (they are instantiated by the compiler and serialized into the output file). That's why you need a constant expression.
Basically you can use all the basic data types (like int
, bool
, string
etc.). You can also use typeof
expressions because they will be resolved to metadata tokens identifying a type, which is fine at compile time.
Maybe you can put the concept of your Func<MyType, bool>
into an interface that your types implement. Or into a separate handler class that you can pass to your Attribute by using a typeof(MyHandlerClass)
expression.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With