Unity's custom == operator

This blog post was posted almost a year ago which made some good arguments for getting rid of the custom == operator for unity objects. Unity 5’s release would have been a great opportunity to make that change since there were other scripting changes needed anyway. Unfortunately, it seems to still be in Unity 5.

So what’s the plan for the == operator now? Will it be removed in a future 5 release or is it sticking around until Unity 6?

On that page:

I don’t think there ever really was a sufficiently compelling reason to cause so much potential script changes, especially since they wouldn’t be able to auto-upgrade these, unlike the changes that actually happened. It definitely will not be removed in Unity 5 since they only make breaking changes with major versions, and I would guess likely not ever.

–Eric

Ah, that’s disappointing. Auto-upgrade would require some proper code inspection but I don’t think it would be too difficult. Microsoft has a code analysis API for that sort of thing.

Yeah, I like the custom implementation. It’s fine once you get used to it. Either way, not a huge deal.

I would have liked removal of implicit casting from Vector3 to Vector2. Again, not a huge deal.

1 Like

I’ve seen someone get fired for overloading operators like that. Ok, maybe that’s a stretch, but I’m pretty sure it contributed to them getting let go before the end of the probation period. Maybe it’s not a “huge deal” but it’s not good coding practice.

It causes confusion which can result in bugs.
I’ve seen plenty of “why doesn’t this code work” questions with the answer being “Because Unity overloads ==”.
And if/when C# 6 becomes available, the .? operator will cause even more confusion.

I understand that it would inconvenience the people are used to it, but I think it’s a bad idea to continue doing something the wrong way just because people are used to it. Either way, it looks like the overloading will continue until at least Unity 6 so there doesn’t seem to be much point in arguing about it right now.

I have to imagine considerations like these change with time.

The Unity guys/gals are probably starting to (or soon will be) discuss Unity 6… They need to at least have a thorough debate on removing the operative overload. This is coming from a team with hundreds-of-thousands of lines of code that will be affected.

I’ve been a .NET developer for over 15 years now, and I cringe at least once a day because of this. It is heinously bad, and casts a long, dark shadow over everything we do in Unity. It has cost us incredible amounts of money in bug hunts and wasted time, and is confusing even for seasoned developers. It causes problems in Asset Store code regularly, and is the source of innumerable forum and Stack questions from new developers.

Whatever fringe benefits exist are absolutely not worth it. Please break our code in Unity 6 and remove it!

2 Likes
void CheckIfNull(ISomeInterface obj) {
    bool isNull = obj == null || (obj is UnityEngine.Object && ((obj as UnityEngine.Object) == null))
    Debug.Log("It is null: " + isNull);
}

:smile:

1 Like

This check function doesn’t really help since it uses the == operator. That’s the purpose for using it then?

It helps. == is a method, and C# uses static dispatch:

IEnumerator Demo() {
    GameObject go = new GameObject("Test Object");
    Destroy(go);
    yield return null;

    object objRef = go;
    Debug.Log(objRef == null); //prints false
    Debug.Log(go == null); //prints true

}

Ok. Now I get it. But it only solves one problem. But the other one is still there…

This isn’t same as

As you will cast obj to UnityEngine.Object then use UnityEngine.Object ==. Maybe you meant (((object)(obj as UnityEngine.Object)) == null)

The original post was a copy from our project. It’s essentially for “check if this reference will throw exceptions if I call methods on it”. So it checks “is this object null in the C# semantics of null, or is this object null in the Unity-semantics of null?”.

Say you have some interface IDamageReceiver:

interface IDamageReceiver {
    void ReceiveDamage(int damage);
}

And some enemy that’s a MonoBehaviour, and is knocked back when it takes damage:

public class Enemy : MonoBehaviour, IDamageReceiver {

    public void ReceiveDamage(int damage) {
        health -= damage;
        transform.Translate(-transform.forward);

        if(health < 0) {
            Die();
        }
    }
}

Then you have some script that does delayed damage to IDamageReceivers:

public class SomeScript : MonoBehaviour {

    public void DamageDelayed(IDamageReceiver dr, float delay) {
        StartCoroutine(DamageAfter(dr, delay));
    }

    private IEnumerator DamageAfter(IDamageReceiver dr, float delay) {
        yield return new WaitForSeconds(delay);
        if(dr != null)
            dr.ReceiveDamage(damage);
    }
}

if the dr that gets sent in is an Enemy, and it has been destroyed at any point, the != null check will pass, and the transform.Translate line will cast a MissingReferenceException.

If I use this code instead:

private IEnumerator DamageAfter(IDamageReceiver dr, float delay) {
        yield return new WaitForSeconds(delay);
        if(!CheckIsNull(dr))
            dr.ReceiveDamage(damage);
    }

bool CheckIsNull(object obj) {
    bool isNull = obj == null || (obj is UnityEngine.Object && ((obj as UnityEngine.Object) == null));
    return isNull;
}

The if-check will not pass, and we won’t get any exceptions. Yay!

1 Like