How do I know if i'm using to many if statements/ when to use a Switch statement.

So i’m on the very early stages of a project i’m working on and fairly new to coding and unity (not pushed a final project but have worked on it for some time)

I am having some concerns right now as well as confused on the best way to call for some code.

A majority of my code is using if statements and since the project is in the early stages I havent seen any slowdown or problems due to the uses of if’s(barely any nested loops). But my major concern is wondering if i’m using to many if statements as well as when is a good time to use a switch statement?

I’m working on an ammo changing method on the gun (thinking I should switch it to the player) and using this file

using UnityEngine;

public class Gun : MonoBehaviour
{
    public bool gunIsHeld;

    [SerializeField]
    private Transform firePoint;
    public GameObject normalBulletPref;
    public GameObject darkBulletPref;
    public GameObject lightBulletPref;
    public float bulletForce = 20;

    //bullet types
    [SerializeField]
    private bool isNormal;
    [SerializeField]
    private bool isDark;
    [SerializeField]
    private bool isLight;

    //gun rotation;
    private Rigidbody2D rb;
    Vector2 mousePos;
    [SerializeField]
    private Transform playerPos;

    public float FireRateMax;
    public float FireRateMin;

    private void Start()
    {
        firePoint = GetComponentInChildren<Transform>();
        rb = GetComponent<Rigidbody2D>();

        FireRateMin = FireRateMax;

        isNormal = true;
    }


    public void Update()
    {
        //checks that the gun is being held by the player;
        IsHeld();

        if(gunIsHeld)
        {
            if (Input.GetMouseButtonDown(1))
            {
                AmmoTypeSwitch();
            }
        }

        if (gunIsHeld)
        {
            transform.position = playerPos.position;

            //start calling for the rotation of the gun before shooting
            mousePos = Camera.main.ScreenToWorldPoint(Input.mousePosition);

            //if left click start shooting
            if (Input.GetMouseButton(0))
            {
                Shoot();
            }
        }
    }

    private void FixedUpdate()
    {
        Vector2 lookDir = mousePos - rb.position;
        float angle = Mathf.Atan2(lookDir.y, lookDir.x) * Mathf.Rad2Deg;
        rb.rotation = angle;
    }

    public void Shoot()
    {
        if(isNormal == true)
        {
            if (FireRateMin == FireRateMax)
            {
                GameObject bullet = Instantiate(normalBulletPref, firePoint.position, firePoint.rotation);
                Rigidbody2D rb = bullet.GetComponent<Rigidbody2D>();
                rb.AddForce(firePoint.right * bulletForce, ForceMode2D.Impulse);

                FireRateMin -= Time.deltaTime * 1;
            }

            else if (FireRateMin != FireRateMax)
            {
                FireRateMin -= Time.deltaTime * 1;

                if (FireRateMin <= 0)
                {
                    FireRateMin = FireRateMax;
                }
            }
        }

        else if(isDark == true)
        {
            if (FireRateMin == FireRateMax)
            {
                GameObject bullet = Instantiate(darkBulletPref, firePoint.position, firePoint.rotation);
                Rigidbody2D rb = bullet.GetComponent<Rigidbody2D>();
                rb.AddForce(firePoint.right * bulletForce, ForceMode2D.Impulse);

                FireRateMin -= Time.deltaTime * 1;
            }

            else if (FireRateMin != FireRateMax)
            {
                FireRateMin -= Time.deltaTime * 1;

                if (FireRateMin <= 0)
                {
                    FireRateMin = FireRateMax;
                }
            }
        }

        else if(isLight == true)
        {
            if (FireRateMin == FireRateMax)
            {
                GameObject bullet = Instantiate(lightBulletPref, firePoint.position, firePoint.rotation);
                Rigidbody2D rb = bullet.GetComponent<Rigidbody2D>();
                rb.AddForce(firePoint.right * bulletForce, ForceMode2D.Impulse);

                FireRateMin -= Time.deltaTime * 1;
            }

            else if (FireRateMin != FireRateMax)
            {
                FireRateMin -= Time.deltaTime * 1;

                if (FireRateMin <= 0)
                {
                    FireRateMin = FireRateMax;
                }
            }
        }
    }

    void AmmoTypeSwitch()
    {
        if (isNormal == true)
        {
            isNormal = false;
            isDark = true;
            isLight = false;

            Debug.Log("Using Dark Ammo");
        }

        else if (isDark == true)
        {
            isNormal = false;
            isDark = false;
            isLight = true;

            Debug.Log("Using light Ammo");
        }

        else if (isLight == true)
        {
            isNormal = true;
            isDark = false;
            isLight = false;

            Debug.Log("Using normal Ammo");
        }
    }


    void IsHeld()
    {
        if (gameObject.transform.parent == null)
        {
            gunIsHeld = false;

            rb.isKinematic = true;
        }
       
        else if(gameObject.transform.parent.CompareTag("Player"))
        {
            gunIsHeld = true;

            rb.isKinematic = false;
        }
    }
}

I’m not sure if if statements will be the best to use in the AmmoTypeSwitch() method or if I should change it to a switch.

The code works but as a fairly new programmer I will still like to learn the best practices so I can grow into a better coder.

P.S. Lastly just wanted to know if it would be better to have the current ammo type on the player and have the gun call from the player to check.

Thank you!

As a fairly new programmer, your first concern should be to finish things and get shot down, everything else comes after. You don’t know what you don’t know, and practice (that mean failing and making mistake) will teach you better.

Generally “if” is unlikely to slowdown your computer in any capacity, not on modern hardware, this isn’t the game boy era anymore. Hierarchy of memory access will be a bigger bottleneck, but at your level it’s probably not even on the map, you probably wouldn’t be pushing high level performance code that soon.

In fact, the process of programming mean that, as you discover the need of your project, you will need to “refactor” the code, ie to re write it to make better and leaner, so don’t get block too early on issue like that, most code are actually ugly anyway, despite the cultural wars that happen in programming forums, most production code, ie on a schedule with deadline, will be “bad”, it’s very easy to find very high profile example from notable company. Also performance is only solved through profiling first and then addressing the actual necessary problem, you get to have the thing work first, THEN you make it performant after profiling, THEN you make it pretty for your ego OR to share with your team.

So at this point “switch” vs “if” is mostly an aesthetics concern, and that depend on the future of your code design too, switch is great if you have a single statement to test, and that in the future you won’t need to break due to exception, it’s a bit more verbose too. If you are in a team just follow the code style, the point isn’t to make code pretty, but to make readable to many in the same group.

Also I guess that worry comes from the stupid piling on yandere simulator code, which cause a lot of misinformation about programming to spread. Don’t listen to that, a proper programming break down has been made by Dyc3 and already disqualify the if vs switch problem.

4 Likes

Like mentioned above, the problem here isn’t with using too many if-statements, and switch-statements wouldn’t make much of a difference.
Rather, there’s a lot of redundancy that’s bloating this script which could be improved.

If you want to refactor some of this code, here’s some things you can do:

if(gunIsHeld)
{
   if(Input.GetMouseButtonDown(1))
   {
      AmmoTypeSwitch();
   }
}

if(gunIsHeld)
{
   transform.position = playerPos.position;

   //start calling for the rotation of the gun before shooting
   mousePos = Camera.main.ScreenToWorldPoint(Input.mousePosition);

   //if left click start shooting
   if (Input.GetMouseButton(0))
   {
      Shoot();
   }
}

No need to check if(gunIsHeld) twice here. Simply move the logic from the second if-statement to the first if-statement.

Next:

mousePos = Camera.main.ScreenToWorldPoint(Input.mousePosition);

Camera.main is actually a shorthand for GameObject.FindWithTag("Main Camera").GetComponent<Camera>();. That’s a fairly expensive operation that you don’t want to run every frame. Instead, cache the reference of Camera.main in the Awake or Start method and use that.

Next:

if (FireRateMin == FireRateMax)
{
   //etc...
}
else if (FireRateMin != FireRateMax)
{
   //etc...
}

else if (FireRateMin != FireRateMax) is an unnecessary extra check, since it is guaranteed to be true if the previous condition of if (FireRateMin == FireRateMax) is false. You can replace that with just an else block instead.

Small one next:

FireRateMin -= Time.deltaTime * 1;

* 1 is an unnecessary mathematical operation here. It will just return the same value as Time.deltaTime anyway. Any value multiplied by 1 will be the same value.

Next:

//bullet types
[SerializeField]
private bool isNormal;
[SerializeField]
private bool isDark;
[SerializeField]
private bool isLight;

You’re trying to structure your code so that only one of these values can be true at any given time. Instead of making separate booleans for each bullet type, create an enum:

[SerializeField]
private BulletType bulletType;

public enum BulletType { Normal, Dark, Light }

Finally, in your Shoot method, you’re repeating the same code multiple times for each condition and just changing the bullet prefab that gets instantiated. You can extract this code into its own method and call it separately instead:

void FireWhenReady(GameObject bulletPrefab)
{
   if (FireRateMin == FireRateMax)
   {
      GameObject bullet = Instantiate(bulletPrefab, firePoint.position, firePoint.rotation);
      Rigidbody2D rb = bullet.GetComponent<Rigidbody2D>();
      rb.AddForce(firePoint.right * bulletForce, ForceMode2D.Impulse);
      FireRateMin -= Time.deltaTime;
   }
   else
   {
      FireRateMin -= Time.deltaTime;

      if (FireRateMin <= 0)
      {
         FireRateMin = FireRateMax;
      }
   }
}

Putting that all together, you’d get something like this:

public class Gun : MonoBehaviour
{
   [SerializeField]
   private BulletType bulletType;

   public GameObject normalBulletPref;
   public GameObject darkBulletPref;
   public GameObject lightBulletPref;

   private Camera mainCam;
   //etc...

   void Start()
   {
      mainCam = Camera.main;
      //etc...
   }

   void Update()
   {
      if(gunIsHeld)
      {
         if(Input.GetMouseButtonDown(1))
         {
            AmmoTypeSwitch();
         }

         transform.position = playerPos.position;

         mousePos = mainCam.ScreenToWorldPoint(Input.mousePosition);

         if (Input.GetMouseButton(0))
         {
            Shoot();
         }
      }
   }

   void Shoot()
   {
      //if-statement would work fine here as well.
      switch(bulletType)
      {
         case BulletType.Normal: FireWhenReady(normalBulletPref); break;
         case BulletType.Dark: FireWhenReady(darkBulletPref); break;
         case BulletType.Light: FireWhenReady(lightBulletPref); break;
      }
   }
 
   void FireWhenReady(GameObject bulletPrefab)
   {
      if (FireRateMin == FireRateMax)
      {
         GameObject bullet = Instantiate(bulletPrefab, firePoint.position, firePoint.rotation);
         Rigidbody2D rb = bullet.GetComponent<Rigidbody2D>();
         rb.AddForce(firePoint.right * bulletForce, ForceMode2D.Impulse);

         FireRateMin -= Time.deltaTime;
      }
      else
      {
         FireRateMin -= Time.deltaTime;

         if (FireRateMin <= 0)
         {
            FireRateMin = FireRateMax;
         }
      }
   }

   void AmmoTypeSwitch()
   {
      if(bulletType == BulletType.Normal)
      {
         bulletType = BulletType.Dark;
      }
      else if(bulletType == BulletType.Dark)
      {
         bulletType = BulletType.Light;
      }
      else
      {
         bulletType = BulletType.Normal;
      }
   }
}

public enum BulletType { Normal, Dark, Light }

And there’s probably still a bit more that can be done here.

Briefly touching on the whole thing with Yandere Simulator brought up, that game’s code issues have been blown up in the wrong way. A lot of people like making those “else if else if else if else if” jokes, but that’s only a surface-level issue that’s easy for people to understand & make fun of. There are problems with the game’s code, but they’re much more significant than just a bunch of if-statements.
Heck, I haven’t seen anyone make any jabs about the 2000+ line long Update method. That would cause significantly more of a performance impact than some if-statements.

2 Likes

Yeah a big reason why I was wondering about the switch vs if statement was because of the Yandere sim problem where everyone says the game was very unoptimized due to running only if statements. If you think this time is about learning and growing from it then I wont worry to much, I’m just working on what I hope to be my first game that I can publish and i’m also in an internship with a small company but feel I’m not learning as much from them as I should be due to it being a team a lesser experienced programmers fighting between who has the better code.

I appreciate the response and i’ll make sure I try to push something out as soon as I can so I can learn from it!

1 Like

Ok thank you for helping clear up the Yandere sim problem and what made me question between if statements and switch statements! as well for the code you shared and helped fix… Kind of mad I didnt see it before because after saying everything and showing me I see all of the problems but hope I get past that bad habit soon! I didnt even realize I called for the if(gunisheld) twice lol! But thank you so much for all of the help and the optimization of the code! I do have some questions though. During the firewhenready method I’ve always been confused with the arguments in the method as well as calling from one code to another. Will you be able to help explain to me how they are used or know a way to help learn them? (when I say code from one to another I mean the var exName = getcomponent(); also, I need help with enum’s but you posted that link and i’ll be sure to look it up!

Again thank you so much for all the help and the optimization!!! I’ll try my hardest to learn from this!!!

1 Like

A lot of it (the jokes) is from one hella 17 000 long (decompiled) script that hold the Whole NPC AI that is able to do investigation based on evidence, with psychological consequence on the NPC, and given the game has a lot of unique character, there is a lot of exception. That code takes less than 1ms to execute after profiling and the slowdown comes from unity’s system (physics, animation, UI, shadow casting), those can be probably optimized, but I feel like it should be on the polish final phases, not a concern during designing the game.

The code really is just a big behavior tree implemented as “if tree”, so it does a lot of skipping in that update() blocks, But you can recognize naive pattern of selector and sequences, thus despite the unique character creating exception, the code is also surprisingly modular, creating character is a matter of appending new behavior and setting a few variable. You can tell that while a non coder, he has some general understanding of AI pattern, his naive behavior tree also have elements of Utility and other pattern. Also it’s so readable people have been modding the game to add behavior without any need for comment, the code is truly transparent.

Also a lot of aesthetics mistakes are artifact of the complexity of a non coder doing one of the most complex social AI for this type of game I had ever seen (only Versu is competing), and designing along the ride, a lot of variable have legacy name whose function has been extended beyond the initial intent, for example the subtree “if(flyAway)” fire when npc see a corpse, but now they don’t just fly away, a lot of unique behavior has been appended, like if the corpse is the bully of the NPC, the npc turn evil and don’t report the corpse, if the witness the player murdering, well they might actually help him.

Similarly a lot of nested empty if, or obvious test that could be skip are artefact of a ongoing design where you test stuff, move and remove code to see if that help the design or not, my codes tend to have those in spade when I’m designing. If anything the biggest misstep, that isn’t superficial aesthetics, is the (lack) structure of his blackboard for the AI, most bug can be trace down to that, that 17000 line of code, 6000 lines is only the code, everything else is variable! Most of them are the black board data. That mean that state management can be a bit sloppy and have mutually exclusive state be true at the same time (like being dead yet move), at the same time it’s also pretty easy to tell what’s broken, nothing is obscure, the naming is straight forward.

That’s about to happen when you grow such a complex and ambitious beast organically and most insight are hindsight, and the game is still not done so you probably will be wrong to choose a structure over another (premature optimizations and all), following the project I have seen many features dropped that would have mess this. Also the naive BT sequence could benefit from being a coroutines. While it could probably broke that script into many chunk, being all in a single place is actually a plus with code folding, the tree nature makes it very easy to navigate once fold, the naming and flow are very clear, and the behavior well ordered (despite some legacy naming due to old design direction).

That was such an interesting thing to analyze personally. I try to read released code of games, but most are too complex or fragmented, this one was a breeze to read, very interesting. I did read it whole …

3 Likes

thats crazy. I heard of people reading the code but the whole thing that must have taken awhile! So i feel better about my code now cause it isn’t all just shoved into the update method. I believe thats what you were trying to get at. the fact that there was so much jumbled up into nested if statements calling to each other in one method!

Thank you for your time! I’ll try to do as much as I can to change the mindset I had and to make sure to stick with my game! I hope to try to publish this game!

Nope I was saying it was fine in the context of that specific dev timeline, team composition, skills and ambition. A lot of AI code bloat, in a typical game, just get hidden into intractable visual tools and spaghetti node, which is just the programmer pushing the responsibility to the designer.

You would have to know what is a behavior tree and how it works to know why this was actually an optimal solution in this case. Like I said “code folding” (when you collapse block of code in the IDE) makes it very manageable, once fold, it’s actually very easy to navigate and is laid out in a very sensible way. Think of the IF structure of yandere as a nested folder structure, even though you have a lot of items, you don’t get to go through all of them, the search is O(log n) so that’s the most efficient way to organize.

For example let’s go back to the example I gave:

The first thing is that it’s not a schedule action, so you don’t check the zillion lines of code of schedule behavior (which has it’s own skips to avoid executing the zillion line of code), that’s one if down, when you check flyaway, you also check if the corpse is teh bully of the current npc AND check if the player is suspicious (ie has blood or hold a weapon), if not check the personality, if coward go to a safe place else go see a teacher. You skip 99% of the code right there to only set a few state on the leaf block. For example you have skip all the code pertaining to the complex psychological fallout of the corpse being the bully of the npc, that won’t ever be executed or tested, and that the most complex part.

On top of that he does the great thing of separating concern, the BT structure don’t execute behavior, it merely set things up for another function down the line to execute. SO adding behavior is separated from the logic, which allow the code to allow both generic behavior and character/group specific behavior. The sophistication of the game wouldn’t have be this complex. Yandere dev find a nice formula where the code is basically the design document, and is easy to edit.

Plus since it’s “low level” coding, the compiler will do further optimization to smooth things out. It works because it’s a single person team, if the team was (way) bigger you would have to develop a tools and modularized the code just a bit more. The irony is that this would be more “proper” but also potentially slower in performance (but probably not enough to start caring) as you would have to have a lot of random memory access that the compiler can’t optimized. So what you gain in production (by allowing multiple designer do behaviors in parallel) you probably lose in performance.

Context matter a lot. I was trying to expose the context and how a real dev can lead you some path. Like I say hindsight is easy. NO project will ever have have perfect code or great architecture, especially from the get go, you can’t always predict the need of the project ahead of time.

Sure thing. It’s a bit of a long post though, but feel free to read if you’re interested.

Method Arguments & Return Types

Method arguments are simply some sets of values that a method requests to run. For example:

void Start()
{
   PrintMessage("Hello");
   PrintMessage("World");
}

void PrintMessage(string message)
{
   //The first time this method is called, "message" = "Hello".
   //The second time this method is called, "message" = "World".
   Debug.Log(message);
}

This PrintMessage method requests a string value as an argument, and will print that value to the Unity console.
In the Start method, we call PrintMessage twice, the first time giving it a value of “Hello”, and the second time, a value of “World”, showing how the value of the method argument (“message”) is assigned when the method is called.

If we go back to the previous post, where we call the FireWhenReady method…

void Shoot()
{
   switch(bulletType)
   {
      case BulletType.Normal: FireWhenReady(normalBulletPref); break;
      case BulletType.Dark: FireWhenReady(darkBulletPref); break;
      case BulletType.Light: FireWhenReady(lightBulletPref); break;
   }
}

void FireWhenReady(GameObject bulletPrefab)
{
   if (FireRateMin == FireRateMax)
   {
      //if bulletType == Normal, "bulletPrefab" = "normalBulletPref".
      //if bulletType == Dark, "bulletPrefab" = "darkBulletPref".
      //if bulletType == Light, "bulletPrefab" = "lightBulletPref".
      GameObject bullet = Instantiate(bulletPrefab, firePoint.position, firePoint.rotation);

      Rigidbody2D rb = bullet.GetComponent<Rigidbody2D>();
      rb.AddForce(firePoint.right * bulletForce, ForceMode2D.Impulse);
      FireRateMin -= Time.deltaTime;
   }
   else
   {
      FireRateMin -= Time.deltaTime;
      if (FireRateMin <= 0)
      {
         FireRateMin = FireRateMax;
      }
   }
}

…The switch statement compares the value of bulletType and passes the corresponding prefab GameObject into the FireWhenReady method. This allows all of the logic inside FireWhenReady to be re-used with different arguments, so that you don’t have to re-write all of the logic again for every different bullet prefab.

In general, if you find that you’re re-writing the same code in multiple places, it can and likely should be turned into a method.

Methods also have what are called return types. If you’re wondering what “void” is when you define a method, that is one such return type. “void” simply means “this method returns nothing”, but sometimes you may want to run a method and return a result after it completes.
A type is basically just the name of an object. “MonoBehaviour”, “Vector3”, “int”, “bool”, “Rigidbody”, are all examples of types, and any custom types you create are valid as well (like your “Gun” script).

Here’s an example:

void Start()
{
   //sum = 6;
   int sum = AddNumbers(2, 4);

   //greaterNumber = 12.087;
   float greaterNumber = GetGreaterNumber(8.125f, 12.087f);

   //Logs a message stating if this GameObject is active or not.
   if(GameObjectIsActive())
   {
      Debug.Log("This GameObject is active.");
   }
   else {
      Debug.Log("This GameObject is inactive.");
   }
}

//This method returns an int type.
int AddNumbers(int num1, int num2)
{
   //Returns the sum of the two method arguments.
   return num1 + num2;
}

//This method returns a float type.
float GetGreaterNumber(float num1, float num2)
{
   float greaterNumber;

   if(num1 > num2) {
      greaterNumber = num1;
   }
   else {
      greaterNumber = num2;
   }

   //Returns the greater number of the two method arguments.
   return greaterNumber;
}

//This method returns a bool type.
bool GameObjectIsActive()
{
   //Returns true if this GameObject is active in the scene, or false if not.
   return gameObject.activeSelf;
}

GetComponent & Generics

This one’s a two-fold example. It should be noted that GetComponent is specifically a Unity-only thing, but the part where you pass a type within the <> symbols is a native feature of C# called Generics, and they’re a bit more of an intermediate-advanced subject.

As previously mentioned, you pass a type within the <> symbols.
When you pass a type into a Generic method/class/interface, it basically means, “do this using this type of object”.

So getting back to GetComponent<>(), what Unity does is search for and return a script attached to a GameObject that is the type specified within the <> symbols.
For instance, when you write Rigidbody rb = GetComponent<Rigidbody>();, you’re telling the method, “Get me a script attached to the GameObject that is a Rigidbody type”, and it will return the first Rigidbody script it could find on the GameObject.

Enums

An enum is just a set of named constant values you can choose between.
Say you’re making a calendar app, where the user can add an event onto a day of the week. Without using enums, you would probably use strings or ints instead to reference the day of the week, however that comes with a problem such as this:

class CalendarApp
{
   void SomeMethod()
   {
      AddCalendarEvent("Monday", "Uncle's Birthday");
      AddCalendarEvent("Wednesday", "Project Deadline");
      AddCalendarEvent("Saturday", "Anniversary");

      //You only want to pass the named days-of-the-week, but a string could be anything.
      //Imagine if the user had to type-in the day of the week in a textfield. They could do something like this:
      AddCalendarEvent("gisdhufiugsdfghlauibsg", "This will run, but it won't work");
   }

   void AddCalendarEvent(string dayOfWeek, string eventName)
   {
      //etc...
   }
}

With an enum though, a defined set of named constants restricts what values can be entered:

enum DayOfWeek
{
   Monday,
   Tuesday,
   Wednesday,
   Thursday,
   Friday,
   Saturday,
   Sunday
}

So if we fix up our calendar app to use this DayOfWeek enum as so…

class CalendarApp
{
   void SomeMethod()
   {
      AddCalendarEvent(DayOfWeek.Monday, "Uncle's Birthday");
      AddCalendarEvent(DayOfWeek.Wednesday, "Project Deadline");
      AddCalendarEvent(DayOfWeek.Saturday, "Anniversary");

      //You can only use the named values inside of "DayOfWeek".
      //This code will not even compile and cannot be run.
      AddCalendarEvent(DayOfWeek.gisdhufiugsdfghlauibsg, "This won't even compile");
   }

   void AddCalendarEvent(DayOfWeek dayOfWeek, string eventName)
   {
      //etc...
   }
}

public enum DayOfWeek
{
   Monday,
   Tuesday,
   Wednesday,
   Thursday,
   Friday,
   Saturday,
   Sunday
}

…The day-of-week inputted by the user is guaranteed to be a valid day.

1 Like

thank you i’ll look over it soon! i’m currently going to make a health system for my character and then try to make it so the character can deal and take damage from the enemies when I make them!