Unable to cast object of type 'System.Object[]' to 'MyObject[]', what gives?

30,984

Solution 1

Alternative answer: generics.

public static T[] CreateProperties<T>(IProperty[] properties)
    where T : class, new()
{
    //Empty so return null
    if (properties==null || properties.Length == 0)
        return null;

    //Check the type is allowed
    CheckPropertyTypes("CreateProperties(Type,IProperty[])",typeof(T));

    //Convert the array of intermediary IProperty objects into
    // the passed service type e.g. Service1.Property
    T[] result = new T[properties.Length];
    for (int i = 0; i < properties.Length; i++)
    {
        T[i] = new T();
        ServiceUtils.CopyProperties(properties[i], t[i]);
    }
    return result;
}

Then your calling code becomes:

Property[] props = ObjectFactory.CreateProperties<Property>(properties);
_service.SetProperties(folderItem.Path, props);

Much cleaner :)

Solution 2

Basically, no. There are a few, limited, uses of array covariance, but it is better to simply know which type of array you want. There is a generic Array.ConvertAll that is easy enough (at least, it is easier with C# 3.0):

Property[] props = Array.ConvertAll(source, prop => (Property)prop);

The C# 2.0 version (identical in meaning) is much less eyeball-friendly:

 Property[] props = Array.ConvertAll<object,Property>(
     source, delegate(object prop) { return (Property)prop; });

Or just create a new Property[] of the right size and copy manually (or via Array.Copy).

As an example of the things you can do with array covariance:

Property[] props = new Property[2];
props[0] = new Property();
props[1] = new Property();

object[] asObj = (object[])props;

Here, "asObj" is still a Property[] - it it simply accessible as object[]. In C# 2.0 and above, generics usually make a better option than array covariance.

Solution 3

As others have said, the array has to be of the right type to start with. The other answers have shown how to convert a genuine object[] after the fact, but you can create the right kind of array to start with using Array.CreateInstance:

object[] result = (object[]) Array.CreateInstance(type, properties.Length);

Assuming type is a reference type, this should work - the array will be of the correct type, but you'll use it as an object[] just to populate it.

Solution 4

You can't convert an array like that - it's returning an array of objects, which is different from an object. Try Array.ConvertAll

Solution 5

That's correct, but that doesn't mean that you can cast containers of type Object to containers of other types. An Object[] is not the same thing as an Object (though you, strangely, could cast Object[] to Object).

Share:
30,984
Rob Stevenson-Leggett
Author by

Rob Stevenson-Leggett

I love to code and snowboard. Follow me on twitter: @rsleggett

Updated on July 05, 2022

Comments

  • Rob Stevenson-Leggett
    Rob Stevenson-Leggett almost 2 years

    Scenario:

    I'm currently writing a layer to abstract 3 similar webservices into one useable class. Each webservice exposes a set of objects that share commonality. I have created a set of intermediary objects which exploit the commonality. However in my layer I need to convert between the web service objects and my objects.

    I've used reflection to create the appropriate type at run time before I make the call to the web service like so:

        public static object[] CreateProperties(Type type, IProperty[] properties)
        {
            //Empty so return null
            if (properties==null || properties.Length == 0)
                return null;
    
            //Check the type is allowed
            CheckPropertyTypes("CreateProperties(Type,IProperty[])",type);
    
            //Convert the array of intermediary IProperty objects into
            // the passed service type e.g. Service1.Property
            object[] result = new object[properties.Length];
            for (int i = 0; i < properties.Length; i++)
            {
                IProperty fromProp = properties[i];
                object toProp = ReflectionUtility.CreateInstance(type, null);
                ServiceUtils.CopyProperties(fromProp, toProp);
                result[i] = toProp;
            }
            return result;
        }
    

    Here's my calling code, from one of my service implementations:

    Property[] props = (Property[])ObjectFactory.CreateProperties(typeof(Property), properties);
    _service.SetProperties(folderItem.Path, props);
    

    So each service exposes a different "Property" object which I hide behind my own implementation of my IProperty interface.

    The reflection code works in unit tests producing an array of objects whose elements are of the appropriate type. But the calling code fails:

    System.InvalidCastException: Unable to cast object of type 'System.Object[]' to type 'MyProject.Property[]

    Any ideas?

    I was under the impression that any cast from Object will work as long as the contained object is convertable?

  • Rob Stevenson-Leggett
    Rob Stevenson-Leggett over 15 years
    Thanks, this works. What's the performance impact of convertall vs. copy?
  • leppie
    leppie over 15 years
    So many times I just fall into this trap with valuetypes, then I remember :)
  • Marc Gravell
    Marc Gravell over 15 years
    ConvertAll will involve a delegate invoke each time, where-as Copy should just use Buffer.BlockCopy under the hoop. So Copy should be quicker. ConvertAll is easier to type (and fine for small-to-mid sizes), and more flexible: you can do non-trivial projections.
  • Marc Gravell
    Marc Gravell over 15 years
    Well, to be pedantic, and Object[] is an Object... so the cast isn't that strange. Everything is an Object in .NET
  • Rob Stevenson-Leggett
    Rob Stevenson-Leggett over 15 years
    Great info! thanks, StackOverflow needs assisted answers. It would be nice to know the type but I'm trying to write as little code as possible. This way I can add more services later without having to write the mapping code again. I would have to write 3 loops in 3 methods currently.
  • Marc Gravell
    Marc Gravell over 15 years
    Also - ConvertAll will cope with conversions rather than casts; Array.Copy can't do this, as there is no guarantee that the original and destination objects are the same size (which is necessary for Buffer.BlockCopy)
  • Jon Skeet
    Jon Skeet over 15 years
    Rob: You're already passing in the type as a parameter to the method, aren't you? Have I misunderstood something?
  • Marc Gravell
    Marc Gravell over 15 years
    Arguably, it returns something a little dodgy for the Length==0 case - I'd return an empty array myself...
  • Jon Skeet
    Jon Skeet over 15 years
    Edited to remove accusation :) And yes, returning null is possibly a bit dodgy - but that's a matter for the OP to decide :)
  • Marc Gravell
    Marc Gravell over 15 years
    lol - no accusation taken; I kinda agreed with the original version ;-p
  • Rob Stevenson-Leggett
    Rob Stevenson-Leggett over 15 years
    This is a much better solution, so switching answer (sorry Marc!).
  • Rob Stevenson-Leggett
    Rob Stevenson-Leggett over 15 years
    Yeah... don't know what I was talking about here!