JavaScript instrukcje warunkowe

Wprowadzenie

"Jeżeli kupisz mi zabawkę to nie będę płakał, w przeciwnym razie zrobię aferę na cały sklep."

Pomijając kwestie wychowawcze (maluch nigdy nie powinien dostać zabawki) powyższa wypowiedź to typowy warunek, który steruje działaniami rodziców.

Pisząc nasze skrypty w zasadzie non stop będziemy stosować podobne warunki, które w naszym przypadku będą sterować przebiegiem naszego programu.


	const nr = prompt("Podaj jakąś liczbę");

	if (nr > 20) { //jeżeli...
		document.write("Liczba jest większa od 20");
	} else { //w przeciwnym razie...
		document.write("Liczba jest mniejsza od 20");
	}
		

Wynik sprawdzenia

Gdy porównujemy ze sobą dwie wartości (tak jak powyżej zmienną nr z liczbą 20), wynikiem takiej operacji zawsze będzie wartość Boolean, czyli true lub false (prawda/fałsz):


	const a = 10;
	const b = 20;

	document.write(b > a); //true
	document.write(b < a); //false
	document.write(b === a); //false
	

	const result = a > b;
	document.write(result); //false
		

	const a = 10;
	const b = 20;

	if (a < b) { //kod się wykona
		document.write("A jest mniejsze od B");
	}
		

	const nr = prompt("Podaj jakąś liczbę");
	if (nr > 5) {
		document.write(`Liczba ${nr} jest większa od 5`);
	}
		

Ostatni listing pokazuje charakterystyczną rzecz, z którą nie raz będziemy się stykać w Javascript. Pod zmienną nr pobraliśmy liczbę, którą wpisze nam w okienko użytkownik. Wartość pobierana z takiego okienka zawsze jest w formacie tekstowym (string). Powyższe równanie będzie miało więc przykładowo postać:


	if ("7" > 5) {
		document.write(`Liczba 7 jest większa od 5`);
	}
		

co równoznaczne jest z tym, że przyrównujemy do siebie tekst z liczbą.

W Javascript możemy sprawdzać ze sobą dowolne typy danych.

W przypadku porównywania stringów działa tak zwane porównywanie leksykograficzne (słownikowo), gdzie teksty porównywane są litera po literze:


	document.write("ab" > "aa"); //true
	document.write("pies" > "kot"); //true
	document.write("abc" > "acc"); //false
	document.write("alicja" > "bela"); //false
	document.write("Marcin" > "Ania"); //true - znaczy lepszy 😏
		

Przy takim porównaniu stringów brane są pod uwagę pozycje liter na tablicy znaków Unicode. Powoduje to, że "a" jest większe od "A", ponieważ znajduje się na dalszej pozycji w takiej tabeli.


	document.write("a" > "A"); //true
	document.write("Kot" > "kot"); //false

	//dlatego warto przed konwersją zrównać wielkość liter
	document.write("alicja" > "Beata"); //true
		

W przypadku porównywania wartości innych typów są one konwertowane do liczb, a następnie porównywane są te liczby:


	document.write("3" > 2); //true bo 3 > 2
	document.write("02" > 3); //false bo 2 > 3 chyba
	document.write("0" == 0); //true

	document.write(true > 2); //false bo true to 1
	document.write(false < 2); //true bo false to 0

	document.write("Ala" > 0); //false bo konwersja "Ala" na liczbę to NaN (NonANumber), a NaN jest mniejsze od każdej liczby
	document.write("Kot" > -Infinity); //false - to samo co powyżej. NaN jest mniejsze od każdej liczby
		

Tyle tylko, że to, że się da, wcale nie oznacza, że powinieneś robić takie porównania...

Polecam na co dzień pisać może i nieco dłuższy, ale i bezpieczniejszy kod. Tworząc więc warunki staraj się konwertować dane wartości do podobnego typu.


	//zamiast
	const nr = prompt("Podaj liczbę z zakresu 1-10");
	if (nr > 5) { ... }

	//napisz
	const nr = Number(prompt("Podaj liczbę z zakresu 1-10"));
	if (nr > 5) { ... }
		

Dzięki takiej manualnej konwersji unikniesz błędów w sytuacjach, które na pierwszy rzut okna wydają się całkowicie prawidłowe:


	const nr = prompt("Podaj liczbę 10");
	if (nr == 10) {
		document.write(nr + nr);
	}
		

W powyższym kodzie w konsoli powinniśmy dostać wynik 20, jednak rezultatem jest 1010. Wynika to z tego, że prompt() (podobnie do kontrolek formularzy) zwraca nam dane w formacie tekstowym. Porównanie z linii 2 nie sprawdza typu a tylko wartość, więc "10" == 10 jest prawdziwe. Kod się wykonuje i w rezultacie dostajemy "10" + "10" co daje nam wynik "1010".

Stąd też polecam stosować porównywanie za pomocą trzech znaków (=== lub !==), który w teście bierze pod uwagę także typ danych. Dzięki temu kod taki zmusza nas do jawnej konwersji danej zmiennej na odpowiedni typ:


	const nr = prompt("Wpisz liczbę 10");
	if (nr === 10) { //nic nie zobaczymy w konsoli bo "10" nie równa się 10
		document.write(nr + nr);
	}
		

	const nr = Number(prompt("Wpisz liczbę 10"));
	if (nr === 10) { //to zadziała
		document.write(nr + nr); //20
	}
		

Tutaj warto wspomnieć, że Webstorm domyślnie każdorazowe użycie podwójnego porównania (==) zaznacza na żółto, co oznacza potencjalny problem. Jeżeli nie chcesz żółtych śmieci na ekranie edytora - stosuj potrójne porównanie 😏 . ...Z takim argumentem nie ma co dyskutować.

Wartości falsy

Tworząc warunki, nie musimy porównywać ze sobą dwóch wartości.
Wartością false staje się każda z poniższych wartości.
Są to tak zwane wartości falsy:


	if (false) { ... }
	if (null) { ... }
	if (undefined) { ... }
	if (0) { ... }
	if (NaN) { ... }
	if ("") { ... }
	if (document.all) { ... }
	

Dla nas oznacza to tyle, że tworząc warunki możemy wykonywać kod w zależności od stanu danej zmiennej, ale też wartości wpisanej ręcznie w nawias:


	const a = 20;
	const b = 0;
	const c = null;

	if (a) { //to się wykona bo a !== 0
		document.write("A ma wartość ", a);
	}
	if (b) { //to się nie wykona bo b === 0
		document.write("A ma wartość ", b);
	}
	if (c) { //to się nie wykona bo null
		document.write("A ma wartość ", c);
	}
	if (false) { //to się nie wykona bo false to false
	}
		

Każda inna wartość daje w rezultacie true. Możemy to sprawdzić konwertując dane wartości na Boolean:


	Boolean(false); //false
	Boolean(null); //false
	Boolean(undefined); //false
	Boolean(0); //false
	Boolean(NaN); //false
	Boolean(""); //false
	Boolean(document.all); //false

	Boolean("Ala"); //true
	Boolean(2); //true
	Boolean(2-2); //false

	const x;
	Boolean(x); //false bo x nie ma wartości czyli undefined
	

Powyższe sprawia, że dość często spotkasz zapisy jak poniżej:


	if (nr) { //kod się wykona jeżeli wartość liczby nr jest różna od falsy
	}

	const txt = "Ala ma kota";
	if (txt.length) { //sprawdzam długość tekstu. Jeżeli większa od 0 to true
		...
	}

	const tab = []
	if (tab.length) { //podobnie sprawdzam długość tablicy
		...
	}
		

Instrukcja if

Powyżej w listingach używałem instrukcji if, która wykonuje dany kod tylko w przypadku, gdy w nawiasach wynikiem będzie prawda:


	const nr = Math.random() * 10;

	if (nr > 5) {
		document.write("Liczba nr jest większa od 5");
	}
		

Dla każdej instrukcji if możemy zastosować zapisy else i else if:


	const nr = Math.random() * 10;

	if (nr >= 5) {
		document.write("Liczba nr jest większa lub równa 5");
	} else {
		document.write("Liczba nr jest mniejsza od 5");
	}
		

	const nr = Math.random() * 10;

	if (nr < 3) {
		document.write("Liczba jest mniejsza od 3");
	} else if (nr <= 6) {
		document.write("Liczba jest mniejsza lub równa 6");
	} else {
		document.write("Liczba jest większa od 6");
	}
		

	const name = "Marcin";

	if (name === "Marcin") {
		document.write("Marciny są fajne");
	} else if (name === "Ania") {
		document.write("Anie są fajne");
	} else if (name === "Radosław") {
		document.write("Radki są fajne");
	} else {
		document.write("Nie wiem kto jest fajny");
	}
		

Co kiedy stosować?

Wszystko zależy od sytuacji.

Czasami wystarczy zwykłe if, czasami musimy posiłkować się else, a czasami else if będzie tym czego potrzebujemy.
Zwróć tylko uwagę, że przy każdej instrukcji if (w tym else if) musimy podać warunek, natomiast else zawsze jest bez warunku.

Dodatkowo takie instrukcje mogą być także zagnieżdżone:


	const x = 1;
	const y = 2;

	if (x > 0) {
		if (y > 0) {
			...//jeżeli x > 0 i y > 0
		}
	}

	//to samo co powyżej
	if (x > 0 && y > 0) {
		...//kod wykonywany jeżeli liczba > 0 i druga_liczba > 0
	}
		

	const checkName = true;
	const name = "Ala";

	if (checkName) {
		if (name === "Ala") {
			document.write("Imię zaczyna się na A");
		}
		if (name === "Beata") {
			document.write("Imię zaczyna się na B");
		}
		if (name === "Monika") {
			document.write("Imię zaczyna się na M");
		}
	}
		

Operator warunkowy

Operator warunkowy (tak zwany ternary operator), to tak naprawdę skrócona wersja warunku if:


	const x = (wyrażenie) ? jeżeli_true : jeżeli_false
		

	const i = 1;

	let number = "";
	if (i > 0) {
		number = "dodatnia";
	} else {
		number = "ujemna";
	}


	//to samo tylko w skróconej wersji
	const number = (i > 0)? "dodatnia" : "ujemna";
		

Przykłady zastosowania:


	const x = 23;
	const isEven = (x % 2 === 0)? "parzysta" : "nieparzysta";
	document.write(isEven); //"nieparzysta"

	const age = 21;
	const status = (age < 18) ? "jesteś za młody" : "zapraszamy na seans";
	document.write(status); //"zapraszamy na seans"

	const name = "Ola";
	document.write( (name === "Ola") ? "Masz na imię Ola" : "Nie masz na imię Ola" ) //"Masz na imię Ola"

	const nr = 10;
	const answer === nr ? "yes" : "no";

	const isMember = true;
	document.write( `Koszt usługi to ${ (isMember ? "2.00" : "10.00") }zł` );
		

Switch

Instrukcja switch jest kolejnym sposobem tworzenia warunków - tym razem na zasadzie przyrównania wyniku do konkretnych wartości.


	switch (wyrażenie) {
		case wartość1:
			//fragment wykonywany gdy rezultat wyrażenia jest równy rezultat1 - potrzebuje break;
			break;
		case wartość2:
			//fragment wykonywany gdy rezultat wyrażenia jest równy rezultat2 - potrzebuje break;
			break;
		default:
			//fragment wykonywany gdy powyższe rezultaty nie są równe rezultatowi wyrażenia - nie potrzebuje break;		
	}
		

Każdy przypadek kończy się słowem break, które kończy wykonywanie instrukcji switch.
Jeżeli pominiemy to słowo, wtedy nawet przy pomyślnym przyrównaniu zostaną wykonane kolejne sprawdzenia, co często może powodować błędy.
Dodatkowo instrukcja switch ma specjalny przypadek default który będzie wybierany, gdy wszystkie inne przypadki będą błędne (odpowiednik else w instrukcji if).


	const number = prompt("Wpisz jakiego masz zwierzaka");

	switch (number) {
		case "pies":
			document.write("Psy są najlepsze");
			break;
		case "kot":
			document.write("Koty są lepsze od psów");
			break;
		case "chomik":
			document.write("Każdy chomik jest super");
			break;
		case "świnka":
			document.write("Świnki są fajowe");
			break;
		default:
			document.write("Jakiś dziwny ten zwierzak");
	}
		

Warto zaznaczyć, że instrukcja switch nie służy do testowania warunków, a do przyrównywania zmiennej do wartości:


	const car = "bmw";
	
	switch (car) {
		case "bmw" : ... break;
		case "fiat" : ... break;
		case "audi" : ... break;
	}

	//poniższe przyrównywanie nie zadziała
	const nr = 5;
	switch (nr) {
		case (nr >= 5):
			document.write("Mało");
		break;
		case (nr > 5 && nr <= 10):
			document.write("Średnio");
		break;
		case (nr > 15) :
			document.write("Dużo");
		break;
	}
		

Jednak i tutaj można przyrównać do warunków. Wystarczy przyrównywać zamiast do zmiennej to do wartości true:


	const nr = 5;
	switch (true) {
		case (nr >= 5):
			document.write("Mało");
		break;
		case (nr > 5 && nr <= 10):
			document.write("Średnio");
		break;
		case (nr > 15) :
			document.write("Dużo");
		break;
	}
		

Zastosowanie tego jest niespotykane i traktował bym to jako ciekawostkę.

Projekt i wykonanie: Ryszard Rogacz© 1999−2024