views:

1768

answers:

5

Take the following :

class A {}

class B : A {}

class C
{
    C() {
        var b = new B();
        Foo(b);
        Foo2(ref b); // <= Compile time error : 
                     // 'The 'ref' argument doesn't match the parameter type
    }

    void Foo(A a) {}

    void Foo2(ref A a) {}  
}

Why does the above compile time error occur ? This happens with both ref and out arguments.

+13  A: 

Because in both cases you must be able to assign value to ref/out parameter.

If you try to pass b into Foo2 method as reference, and in Foo2 you try to assing a = new A(), this would be invalid.
Same reason you can't write:

B b = new A();
maciejkow
Ninj'd by 4 seconds! :)
CannibalSmith
+1 Straight to the point and explains perfectly well the reason.
Rui Craveiro
+2  A: 

Because giving Foo2 a ref B would result in a malformed object because Foo2 only knows how to fill A part of B.

CannibalSmith
+76  A: 

=============

UPDATE: I used this answer as the basis for this blog entry:

http://blogs.msdn.com/ericlippert/archive/2009/09/21/why-do-ref-and-out-parameters-not-allow-type-variation.aspx

See the blog page for more commentary on this issue. Thanks for the great question.

=============

Let's suppose you have classes Animal, Mammal, Reptile, Giraffe, Turtle and Tiger, with the obvious subclassing relationships.

Now suppose you have a method void M(ref Mammal m). M can both read and write m. Can you pass a variable of type Animal to M? No. That variable could contain a Turtle, but M will assume that it contains only Mammals. A Turtle is not a Mammal.

Conclusion 1: Ref parameters cannot be made "bigger". (There are more animals than mammals, so the variable is getting "bigger" because it can contain more things.)

Can you pass a variable of type Giraffe to M? No. M can write to m, and M might want to write a Tiger into m. Now you've put a Tiger into a variable which is actually of type Giraffe.

Conclusion 2: Ref parameters cannot be made "smaller".

Now consider N(out Mammal n).

Can you pass a variable of type Giraffe to N? No. N can write to n, and N might want to write a Tiger.

Conclusion 3: Out parameters cannot be made "smaller".

Can you pass a variable of type Animal to N?

Hmm.

Well, why not? N cannot read from n, it can only write to it, right? You write a Tiger to a variable of type Animal and you're all set, right?

Wrong. The rule is not "N can only write to n". The rules are, briefly:

1) N has to write to n before N returns normally. (If N throws, all bets are off.)

2) N has to write something to n before it reads something from n.

That permits this sequence of events:

  • Declare a field x of type Animal.
  • Pass x as an out parameter to N.
  • N writes a Tiger into n, which is an alias for x.
  • On another thread, someone writes a Turtle into x.
  • N attempts to read the contents of n, and discovers a Turtle in what it thinks is a variable of type Mammal.

Clearly we want to make that illegal.

Conclusion 4: Out parameters cannot be made "larger".

Final conclusion: Neither ref nor out parameters may vary their types. To do otherwise is to break verifiable type safety.

If these issues in basic type theory interest you, consider reading my series on how covariance and contravariance work in C# 4.0:

http://blogs.msdn.com/ericlippert/archive/tags/Covariance+and+Contravariance/default.aspx

Eric Lippert
+1 for use of giraffes. =]
Ed Woodcock
+1. Great explanation using real-world class examples that clearly demonstrate the issues (i.e. - explaining with A, B and C makes it harder to demonstrate why it doesn't work).
Grant Wagner
I feel humbled reading this thought process. I think I better get back to the books!
objektivs
In this case, we really cant use Abstract class variable as arguments and pass on its derived class object!!
Prashant
Eric, I'm honored that you are using my question as one of your fantastic blog posts. Thanks.
Andreas Grech
+4  A: 

You're struggling with the classic OOP problem of covariance (and contravariance), see wikipedia: much as this fact may defy intuitive expectations, it's mathematically impossible to allow substitution of derived classes in lieu of base ones for mutable (assignable) arguments (and also containers whose items are assignable, for just the same reason) while still respecting Liskov's principle. Why that is so is sketched in the existing answers, and explored more deeply in these wiki articles and links therefrom.

OOP languages that appear to do so while remaining traditionally statically typesafe are "cheating" (inserting hidden dynamic type checks, or requiring compile-time examination of ALL sources to check); the fundamental choice is: either give up on this covariance and accept practitioners' puzzlement (as C# does here), or move to a dynamic typing approach (as the very first OOP language, Smalltalk, did), or move to immutable (single-assignment) data, like functional languages do (under immutability, you can support covariance, and also avoid other related puzzles such as the fact that you cannot have Square subclass Rectangle in a mutable-data world).

Alex Martelli
+1  A: 

Isn't that the compiler telling you it would like you to explicitly cast the object so that it can be sure you know what your intentions are?

Foo2(ref (A)b)
dlamblin