Why isn't RaycastHit nullable and turned into the return value of Physics.Raycast?

As it is Physics.Raycast returns a bool, and you have the option to add an out parameter to get the RaycastHit variable information.

Why?

You could remove the out parameter entirely and just return RaycastHit which seems cleaner, making the RaycastHit class/struct nullable to keep the ability to use Physics.Raycast directly in if calls

It allows for really clean if statements:

if (Physics.Raycast(ray, out RaycastHit result)) {
  // got a hit!
}

if you returned a nullable RaycastHit you would have to do something a bit more complicated:

var result = Physics.NullableRaycast(ray);
if (result.HasValue) {
  // we got a hit!
}

There may also be memory management advantages - you can reuse the same RaycastHit struct over and over again with the “out” formulation. Feel free to write your own wrapper if you would prefer it though:

public static RaycastHit? RaycastWrapper(Ray ray) {
  if (Physics.Raycast(ray, out RaycastHit result)) {
    return result;
  }
  return null;
}
2 Likes

Oh, I was under the assumption that if(null) == false (and trying it now I see that very few variables are nullable, maybe I’m confusing with C)
I got confused into thinking this way because this works:

GameObject object;
Ray ray = new Ray(Vector3.one, Vector3.one);
RaycastHit hitInfo;

Physics.Raycast(ray, out RaycastHit result);

if(object)
  print("the object exists!");
if(hitInfo.collider)
  print("the raycast hit a collider!");

I was assuming that object and hitInfo.collider were set to null before being set to something.
If it isn’t that, how do they work as both bools and their respective types and why can’t this be the same for RaycastHit?

The reason it works in your example above is because Unity has provided a custom operator for its base object class that allows automatic conversion to bool: Unity - Scripting API: Object.bool

They are taking advantage of a builtin C# feature to define this automatic conversion: User-defined explicit and implicit conversion operators - provide conversions to different types - C# reference | Microsoft Learn

Since RaycastHit is not an object but a struct (which is a value type, not a reference type), it doesn’t work for that, nor does it work for anything that doesn’t extend from UnityEngine.Object

1 Like