添加链接
link管理
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接
Collectives™ on Stack Overflow

Find centralized, trusted content and collaborate around the technologies you use most.

Learn more about Collectives

Teams

Q&A for work

Connect and share knowledge within a single location that is structured and easy to search.

Learn more about Teams

I am getting this error on a routine which uses reflection to dump some object properties, something like the code below.

MemberInfo[] members = obj.GetType().GetMembers(BindingFlags.Public | BindingFlags.Instance) ;
foreach (MemberInfo m in members)
    PropertyInfo p = m as PropertyInfo;
    if (p != null)
       object po = p.GetValue(obj, null);

The actual error is "Exception has been thrown by the target of an invocation" with an inner exception of "Method may only be called on a Type for which Type.IsGenericParameter is true."

At this stage in the debugger obj appears as

  {Name = "SqlConnection" FullName = "System.Data.SqlClient.SqlConnection"} 

with the type System.RuntimeType

The method m is {System.Reflection.MethodBase DeclaringMethod}

Note that obj is of type System.RuntimeType and members contains 188 items whereas a simple typeof(System.Data.SqlClient.SqlConnection).GetMembers(System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.Instance) only returns 65.

I tried checking isGenericParameter on both obj and p.PropertyType, but this seems to be false for most properties including those where p.GetValue works.

So what exactly is a "Type for which Type.IsGenericParameter is true" and more importantly how do I avoid this error without a try/catch?

That means it is a generic type argument in an open generic type - i.e. where we haven't picked a T yet; for example:

// true
bool isGenParam = typeof(List<>).GetGenericArguments()[0].IsGenericParameter;
// false (T is System.Int32)
bool isGenParam = typeof(List<int>).GetGenericArguments()[0].IsGenericParameter;

So; have you got some open generics hanging around? Perhaps if you can give an example of where you got your obj from?

From what I can see, the problem isn't so much the existence of open generics, but the existence of normal types :) – Eric Smith Aug 28, 2009 at 11:34 I start with a exception (in this case a SqlException) which is passed to my dump routine. The dump routine is recursive, so the dump is called on all the properties and fields in the exception. There are a few generics in my code, but no open generics. What is confusing me is the error seems to say that p.GetValue only works if GenericParameter is true which is plainly not true. – sgmoore Aug 28, 2009 at 11:47 @sgmoore: What do you thing GetValue is doing? Its invoking the get operation of property described by the PropertyInfo on the obj. The PropertyInfo describes the DeclaringMethod of a Type object, hence GetValue calls the get operation of DeclaringMethod which is then failing because obj is not a Generic Type of a Method. – AnthonyWJones Aug 28, 2009 at 12:41 @anthonywjones. I understand that (now). It seems obvious in hindsight, but it wasn't until I read Eric Smith's answer that it clicked. My only excuse is that the word Method in the error message threw me. (And yes, I understand the properties are effectively a get and set method, but I don't think of properties as methods and assumed the method it was talking about must be the GetValue method) – sgmoore Aug 28, 2009 at 13:38

Firstly, you've made an incorrect assumption, that is, you've assumed that members has returned the members of an instance of System.Data.SqlClient.SqlConnection, which it has not. What has been returned are the members of an instance of System.Type.

From the MSDN documentation for DeclaringType:

Getting the DeclaringMethod property on a type whose IsGenericParameter property is false throws an InvalidOperationException.

So... it's understandable that an InvalidOperationException is being thrown, since naturally, you're not dealing with an open generic type here. See Marc Gravell's answer for an explanation of open generic types.

I thinking I am beginning to see the light. It isn't saying that p.GetValue can 'only called on a Type for which Type.IsGenericParameter is true' but rather that the underlying property represented by p, which in this case is DeclaringMethod, can only be called is Type.IsGenericParameter is true. – sgmoore Aug 28, 2009 at 12:00 Exactly - that's what "Exception has been thrown by the target of an invocation" means, the "target of an invocation" in this case is the DeclaringMethod property getter, and you'd get the same IsGenericParameter exception by reading obj.DeclaringMethod directly. – stevemegson Aug 28, 2009 at 13:02 I've marked this answer as my accepted answer as it was the most helpful, but in truth most of the other answers helped as well. So thanks to everyone. – sgmoore Aug 28, 2009 at 13:26

All the clues are in there. The type of the obj is the Type class itself (or rather the strange RuntimeType derivative).

At the point of failure you loop has arrived the Type class property called DeclaringMethod. However the type that this instance of the Type class is describing is System.Data.SqlClient.SqlConnection which is not a Generic Type of a method.

Hence attempting to invoke the get on DeclaringMethod results in the exception.

The key is you are examining the type of the class Type. Its a bit circular but think of this:-

SqlConnection s = new SqlConnection();
Type t = s.GetType()
Type ouch = t.GetType()

What is class ouch describing?

How do I avoid this error without a try/catch?

You almost certainly can't. When you call p.GetValue, you're calling the getter on that property, which could throw any kind of exception. For example, SqlConnection.ServerVersion will throw an exception if the connection is closed, and you have to handle that.

Where are these extra members coming from?

Your obj already contains the RuntimeType object representing SqlConnection, rather than an instance of SqlConnection. obj.GetMembers() would return the 65 members of the SqlConnection class, but by calling GetType() again, you get the 188 members of RuntimeType.

What is IsGenericParameter?

Instead of representing a class, you can have an instance of RuntimeType that represents a generic parameter to a class or method (The T and TOutput in List<T>.ConvertAll<TOutput>. In this case, DeclaringMethod on the object representing TOutput would let you get a MethodInfo object representing the ConvertAll<> method. However, when the RuntimeType represents a class, the idea of a declaring method makes no sense. That's why reading the property causes the exception that you saw.

Thanks for contributing an answer to Stack Overflow!

  • Please be sure to answer the question. Provide details and share your research!

But avoid

  • Asking for help, clarification, or responding to other answers.
  • Making statements based on opinion; back them up with references or personal experience.

To learn more, see our tips on writing great answers.