When doing an upcast or downcast, what does really happen behind the scenes? I had the idea that when doing something as:
string myString = \"abc\";
object m
I had the idea that actually no casting code would be embedded in the code itself.
An interesting idea. How did you imagine that this worked?
try
{
object x = 123;
object y = (string)x;
}
catch(InvalidCastException ex)
{ ... }
If the cast produces no code then where does the code that throws the exception happen?
Remember, the primary purpose of a cast from a less specific type to a more specific type is to perform a runtime type check.
Once the type check passes, then sure, nothing else really has to happen. The bits of the reference before the type check and the bits after the type check are the same bits; we've just had the runtime verify that the new usage of the old bits is justified.
if you try to use a string on an Form instance, it will throw an exception of wrong usage (because it detects a Form is not a string or any of its subtypes).
Where does it detect that? I mean, in exactly which instruction is that detected? In the castclass instruction. That's what the castclass instruction is for.
what would happen had I ommited the castclass string line and tried to run the code?
The type safety verifier would have rejected your program. Had you forced the CLR to run it without passing verification then it would have had undefined behaviour. It might have succeeded, it might have failed, it might have formatted your hard disk.
Does it really create a new reference?
Remember, at the implementation level a reference is just a pointer-sized integer. It's a number that the memory manager can use to track the position of the referred-to data. It might be a pointer, it might be a handle, it doesn't matter what it is; it's something that implements the abstract notion of a reference.
When you have a variable that contains 12 and you "replace" its contents with 12, is that a "new" 12 that has just been created or is it the "old" 12? Suppose you make a second variable and put 12 in it too by copying from the first variable. Is that a "new" 12 or the "old" 12? How can you tell? It's a difference that makes no difference. When you make a "new" reference that is identical to an "old" reference is that creating something new? The question is a philosophical question, not a technical one.