Unity's magic null check not properly documented

Unity has overriden the == operator such that “x == null” means “is x null or destroyed”. While there has been plenty discussion about it on the forums, and on the blog, the documentation for == doesn’t capture the whole picture:

That quote is correct, but doesn’t actually explain what’s going on, and is not enough to explain to the user what’s happening.

What we need to know is something along the lines of:
“Unity has overridden the == operator for UnityEngine.Object. When you compare and object with null, the check will return true either if the object is null, or if it has been Destroyed.”. A link to an explanation of how objects live both on the c++ side and the C# side, and how this weirdness falls out of the disconnect between those two would also be helpful.

This == null thing isn’t just a minor feature - this is a huge, fundamental change to how the language works. It will surprise any programmer that has written in an object oriented language before they start using Unity. This method needs a clear and long-form explanation of exactly what == does differently from every other C# framework.

As for the example given, the real caveat that people will run into is not mistakenly comparing a new UnityEngine.Object with null, as the documentation example proposes. I doubt that will ever happen - why would anyone ever try to make a blank Object? What will happen is that you’ll get inconsistent == null checks when handling references of an interface type, which experienced programmers will do a bunch. Here’s a quick example of something where code hidden behind a null-check will throw a nullReferenceException:

public interface SomeInterface {
    Vector3 GetPosition();
}

public class SomeBehaviour : MonoBehaviour, SomeInterface {

    public Vector3 GetPosition() {
        return transform.position;
    }
   
}

//Example where this goes wrong:
public static void PrintPosition(SomeInterface behaviour) {
    /*
     * Since behaviour is sent in as a SomeInterface, this == resolves to the default C# ==, not the
     * one defined in UnityEngine.Object.
     */
    if (behaviour == null) {
        Debug.Log("It is null!");
    }
    else {
        /* If behaviour is a SomeBehaviour that has been destroyed, this will throw a NullReferenceException
         * when SomeBehaviour.GetPosition tries to access the transform component
         */
        Debug.Log(behaviour.GetPosition());
    }
}

That’s an example of an actual problem users will run in to.

1 Like

OK, I see what you are saying. Thanks for the report! I’ve added it to our buglist.

Wow, this is nasty. I had no idea something like this could be floating around waiting to cause problems in our code base! @Alex_May just to clarify when your said you reported it, do you mean you filed a report to update the documentation, or to “fix” this issue?

Since the fix would break every_single_project written in Unity, it probably won’t happen unless there’s a major version change (5->6).

It would also require the introduction of an IsDestroyed method, and a retraining of all Unity devs to use it. While I’d prefer that, I don’t see this as a huge problem. The problem is that Unity’s underestimating how huge of a caveat this is, and that it’s not documented properly.

All of Unity’s tutorial resources is geared towards people who’s got very little, if any, programming experience. There’s no “here’s what somebody who knows how to program need to know” introduction, so all of the important details gets hidden away in the docs for specific parts of the API.

1 Like

The former.

We’ve faced some situations (2 at least) when Unity silently hits into null ref and doesn’t throw anything into Console.