Instrukcja przełącznika (Switch) zapewnia bardziej elegancki sposób sprawdzania zmiennej dotyczącej równości względem listy wartości.
Każda wartość jest nazywana przypadkiem (case), a wybrana zmienna, jest sprawdzana dla każdego przypadku przełącznika.
Każdy przypadek reprezentuje wartość do sprawdzenia, następnie dwukropek i instrukcja, które ma zostać wykonana, jeśli przypadek jest spełniony.
Console.WriteLine("Rozbrajasz bombę, który kabel chcesz przeciąć: \n" + "1. zielony \n"+ "2. czerwony \n" + "3. niebieski"); string kabel = Console.ReadLine(); switch (kabel) { case "1": Console.WriteLine("Booom!"); break; case "2": Console.WriteLine("Bomba rozbrojona!"); break; case "3": Console.WriteLine("Booom!"); break; default: Console.WriteLine("Majstrowałeś przy bombie nie przy kablach, " + "bomba zrobiła BOOOOM!"); break; }
Ważną rolę odgrywa tutaj wyrażenie break, które kończy instrukcję switch, zapobiegając sprawdzaniu innych jej przypadków. Bez tego wykonywanie instrukcji jest kontynuowane i przejdzie do następnego bloku instrukcji, nawet jeśli etykieta przypadków nie pasuje do zmiennej typu switch. Współczesny kompilatorom C # nie skompiluje takiego kodu. Wszystkie przypadki i kod deafault muszą kończyć się słowem break lub goto.
Jaki w ogóle jest więc sens tworzenia przełączników, skoro można to samo osiągnąć, stosując instrukcję if? Pierwszym bardzo ważnym elementem jest estetyka, przełącznik jest zwykle bardziej zwarty niż if… else…, a tym samym bardziej czytelny.
Console.WriteLine("Rozbrajasz bombę, który"+ " kabel chcesz przeciąć: \n" + "1. zielony \n"+ "2. czerwony \n" + "3. niebieski"); string kabel = Console.ReadLine(); switch (kabel) { case "1": Console.WriteLine("Booom!"); break; case "2": Console.WriteLine("Bomba rozbrojona!"); break; case "3": Console.WriteLine("Booom!"); break; default: Console.WriteLine("Majstrowałeś przy"+ " bombie nie przy kablach, " + "bomba zrobiła BOOOOM!"); break; } |
Console.WriteLine("Rozbrajasz bombę, który"+ " kabel chcesz przeciąć: \n" + "1. zielony \n"+ "2. czerwony \n" + "3. niebieski"); string twojWiek = Console.ReadLine(); if (twojWiek == "1") { Console.WriteLine("Booom!"); } else if (twojWiek == "2") { Console.WriteLine("Bomba rozbrojona!"); } else if (twojWiek == "3") { Console.WriteLine("Booom!"); } else { Console.WriteLine("Majstrowałeś przy" + " bombie nie przy kablach, " + "bomba zrobiła BOOOOM!"); } |
Drugi to taka, że przełącznik działa znacznie szybciej niż równoważna klatka if-else. To dlatego, że kompilator generuje tabelę skoku dla przełącznika podczas kompilacji. W konsekwencji podczas wykonywania zamiast sprawdzania, która z nich jest spełniona, decyduje tylko o tym, które działanie ma zostać wykonane. Zamiast „break;”, możemy zakończyć przypadek za pomocą „goto case <…>;” dzięki temu, możemy wywołać więcej niż jedną instrukcję przełącznika, wywołać kilka naraz lub przenieść się do wybranej instrukcji nawet tej poza przełącznikiem. Poniższy kod prezentuje pierwszą prostą pętle, która zatrzymuje się w momencie, gdy napisów BOOM! jest >= 100:
Console.WriteLine("Rozbrajasz bombę, który kabel chcesz przeciąć: \n" + "1. zielony \n"+ "2. czerwony \n" + "3. niebieski"); int stoBOOM = 0; string twojWiek = Console.ReadLine(); switch (twojWiek) { case "1": if (stoBOOM >= 100) { Console.WriteLine("Sto razy Booom Booom!"); break; } else { Console.WriteLine("Booom!"); stoBOOM++; goto case "3"; } case "2": Console.WriteLine("Bomba rozbrojona!"); break; case "3": Console.WriteLine("Booom!"); goto case "1"; default: Console.WriteLine("Majstrowałeś przy bombie nie przy kablach, " + "bomba zrobiła BOOOOM!"); break; }
Wykonując powyższy kod, gdy użytkownik wybierze niewłaściwy kabel, przełącznik skacze z przypadku 3 na 1 i z 1 na 3 ponad sto razy, spowodowane jest to odwołaniem do określonego przypadku, a nie wykorzystanie słowa zatrzymującego instrukcję. Drugim sposobem wykorzystania goto jest odwołanie do etykiety. Zapis nie różni się za bardzo, wykorzystując etykiety, pomijamy słowo „case”, nie przejmując się już rodzajem zmiennej warunku.
int k = 0; switch (k) { case 0: Console.WriteLine("k jest równe: " + k); goto dlaWszystkich; case 1: Console.WriteLine("k jest równe: " + k); goto dlaWszystkich; dlaWszystkich: Console.WriteLine("VisualMonsters.cba.pl"); break; } //Otrzymamy: //k jest równe: 0 //VisualMonsters.cba.pl
Jak widzimy, nadałem zamiast „default:” przypadek „dlaWszystkich:” (etykieta ostatniego przypadku), do którego odwołanie jest we wszystkich przypadkach, ten konkretny przypadek kończy całą instrukcję i dopisuje tekst do wyniku.
Etykiety sprawiają, że nasz kod jest jeszcze bardziej czytelny. Dodatkowo etykiety dają nam możliwość skoku nie tylko wewnątrz przełącznika, możemy je umieścić w konkretnej linii kodu i wywołać do nich skok:
int k = 1; switch (k) { case 0: Console.WriteLine("k jest równe: "+k); goto dlaWszystkich; case 1: Console.WriteLine("k jest równe: " + k); goto line33; dlaWszystkich: Console.WriteLine("VisualMonsters.cba.pl"); break; } Console.WriteLine("Tego nie wyswietli"); line33:; Console.WriteLine("To już wyświetli"); //Otrzymamy: //k jest równe: 1 //To już wyświetli
Kod poniżej imituje pewną pętlę, która zmusza użytkownika do wpisania imienia:
NieWpisalesImienia:; Console.Write("Podaj swoje imię: "); string imie = Console.ReadLine(); switch (imie) { case "": Console.WriteLine("Nie wpisałeś imienia, spróbój jeszcze raz. "); goto NieWpisalesImienia; default: Console.WriteLine("Witaj " + imie); break; }
Jeśli użytkownik nic nie wpisze, skok wykonywany jest na początek kodu. Oczywiście skoki można wykorzystać również w instrukcji if, ale wykorzystując je w instrukcji switch, można wydobyć ich prawdziwy potencjał. Wykonując skok, możemy pominąć pewne obszary kodu, co powoduje, że nie są one w ogóle wykonywane, a kod realizowany jest szybciej (ponieważ pomijamy elementy, które mogą go spowolnić).