IB113 Radek Pelánek 2024 Rozcvička I a = [3, 1, 7] print(sorted(a)) print(a) b = [4, 3, 1] print(b.sort()) print(b) 2/72 Rozcvička a = ["magic"] a.append(a) print (a [1] [1] [1] [1] [1] [0] [1] [0] [0] [0]) „záludnější partie" • globální a lokální proměnné • reprezentace dat v paměti, kopírování • předávání parametrů funkcím • typy • rekurze základní témata obecně důležitá detaily specifické pro Python Dnešní programy Varování: Dnešní ukázky programů jsou často ,, • minimalistické ukázky důležitých jevů • nikoliv pěkný, prakticky používaný kód Python Tutor - vizualizace Vizualizace použité ve slidech: http://www.pythontutor.com Odkazy na dnešní ukázky: https://www.fi.muni.cz/~xpelanek/IB113/tutor-codes.html Doporučeno projít si interaktivně. 6/72 Názvy proměnných - konvence • konstanty - velkými písmeny • běžné proměnné: • smysluplná slova • víceslovné názvy: lower_case_with_undersc • krátké (jednopísmenné) názvy: • indexy • souřadnice: x, y • pomocné proměnné s velmi lokálním využitím Globální a lokální proměnné Globální proměnné • definovány globálně (tj. ne uvnitř fun • jsou viditelné kdekoli v programu Lokální proměnné • definovány uvnitř funkce • jsou viditelné jen ve své funkci Rozsah proměnných obecněji • proměnné jsou viditelné v rámci svého „rozsahu" • rozsahem mohou být: • funkce • moduly (soubory se zdrojovým kódem) • třídy (o těch se dozvíme později) • a jiné (závisí na konkrétním jazyce) relevantní terminologie: „namespace", „scope" 9/72 Globální a lokální proměnné a = "This is global." def examplel(): b = "This is local." print(a) print(b) examplel() # This is global. # This is local. print(a) # This is global. print(b) # ERROR! # NameError: name 'b' is not defined Globální a lokální proměnné Python 3.6 1 a = "This is global." 2 def examplel(): b = "This is local pri nt(a) print(b) 3 4 Í 6 7 examplel () 3 print(a) 11 print(b) # This is global # This i s local. # This is global # ERROR! Edit code I Live programming Print output (drag lower right corner to resize) This is global Fra rnes Global frame examplel This is global." examplel b "This is local Objects function examplel() 1 Globální a lokální proměnné vytváříme novou lokální proměnnou, neměníme tu globální a = "Think global." def example2(): a = "Act local." print(a) print(a) # Think global. example2() # Act local. print(a) # Think global. 12/72 Globální a lokální proměnné Python 3.6 a = "Think global." 2 3 def example? (): a = "Act local." print(a) 6 print(a) # Think global. example2() # Act local, print(a) # Think global. Edit code | Live programming st executed jcute > set a breakpoint; use the Back and Forward buttons to jump there. Print output (drag lower right corner to resize) Think global. Frames Global frame example2 example2 Think global Objects function example2() a "Act local. 13/72 Globální a lokální proměnné Jak měnit globální proměnné? a = "Think global." def example3(): global a a = "Act local." print(a) print(a) # Think global example3() # Act local. print(a) # Act local. Lokální proměnné: deklarace lokální proměnná vzniká, pokud je přiřazení kdekoli uvnitř f u n kce a = "Think global, def example4(change_opinion=False): print(a) if change_opinion: a = "Act local." print("Changed opinion:", a) print(a) # Think global. example4() # ERROR! # UnboundLocalError: local variable 'a' reference # assignment < □ ► type(3.0) # type(3==0) # type("3") # type ([3]) # type((3,0)) # type({3:0}) # type({3}) # 40/72 • neměnitelné (immutable): bool, int, float, str, tuple • měnitelné (mutable): list, diet, set Příklady, kde důležité: • změna indexováním • předávání parametrů funkcím o indexování slovníku 41/72 • None - jedinečná hodnota typu NoneType • význam: „prázdné", „žádná hodnota" • využití: např. defaultní hodnota parametrů funkcí • implicitní návratová hodnota z funkcí (pokud nepoužijeme return) 42/72 ravdivostnŕ hodnota if value: print("foo") Pro které z těchto hodnot value se vypíše f < True, False, 3, 0, 3.0, -3, [3], [] , None ravdivostnŕ hodnota test je úspěšný ("tme") vždy, kromě následujících prípadu • konstanty: None, False • nulové hodnoty u numerických typů: 0, 0.0, Oj • prázdne sekvence (nulová délka měřená pomocí len) D. (). {} (mírně zjednodušeno, např. u objektů může být komplikovanější) • type(x) - zjištění typu • isinstance(x, t) - test, zdaje proměnná určitého typu values = [3, 8, "deset", 4, "dva", "sedm", 6] s = 0 for value in values: if isinstance(value, int): s += value else: print("Not int:", value) print("Sum of ints:", s) 45/72 • použití funkce při její vlastní definici • volání sebe sama (s jinými parametry) To iterate is human, to recurse divine. (L. Peter Deutsch) 46/72 xkcd: Tabletop Roleplaying YOUR PARTY ENTERS THE TAVERN. I GATHER EVERYONE AROUND / A TABLE. I HAVE THE ELVE5 / START WHITTLING PICE AND / GET OUT SOME RURCHMENT / FOR CHARACTER 5HEETS. / \ HEY, NO RECURSW. \ I https://xkcd.com/244/ 47/72 https://xkcd.com/688/ 48/72 Sebereferenčnř test O Které písmeno není správnou odpovědí na žádnou otázku: (A) A (B) C (C) B O Odpověď na 3. otázku je: (A) stejná jako na 1. otázku (B) stejná jako na 2. otázku (C) jiná než odpovědi na 1. a 2. otázku O První otázka, na kterou je odpověď A, je otázka: (A)l (B)2 (C)3 Hlavolamikon 49/72 Rekurze a sebereference Rekurze a sebereference - klíčové myšlenky v informatice některé souvislosti: • matematická indukce • funkcionální programování • rekurzivní datové struktury (např. stromy) • gramatiky • logika, neúplnost • nerozhodnutelnost, diagonalizace 50/72 Rekurze a sebereference ... nejen v informatice M. C. Escher; Drawing Gödel, Escher, Bach: An Eternal Golden Braid; Douglas Hofstadter Faktoriá n! = 1 • 2 • • • (n — 1) • n f{n) = 1 pokud n = 1 n • f (n — 1) pokud n > 1 52/72 Faktoriál iterativně (pomocí cyklu) def fact(n): f = 1 for i in range(1,n+l): f = f * i return f 53/72 Faktoriál rekurzivně def fact(n): if n == 1: return 1 else: return n * fact(n-l) 54/72 Faktoriál rekurzivně - ilustrace výpočtu fact (4) 24 4 * fact(3 3 * fact (2 '2 ftctd) 1 55/72 Vymyslete funkci, která: • vypíše čísla od 1 do N • pomocí rekurze - bez použití cyklů f or, while 56/72 Vymyslete funkci, která: • vypíše čísla od 1 do N • pomocí rekurze - bez použití cyklů for, while def sequence(n): if n > 1: sequence(n-1) print(n) 57/72 Co udělá tento program? def test(n): print(n) if n > 1: test(n-1) print(-n) test(5) 58/72 Ilustrace zanořování test(3) 3 test(2) def def a rekurze even(n): print("even", n) odd(n-l) odd(n) : print("odd", n) if n > 1: even(n-l) even(lO) 60/72 61/72 Rekurzivní stromeček nakreslit stromeček znamená: • udělat stonek • nakreslit dva menší stromečky (pootočené) • vrátit se na původní pozici Stromeček želví grafikou def tree(length): forward(length) if length > 10: left(45) tree(0.6 * length) right(90) tree(0.6 * length) left(45) back(length) 63/72 oby rekurze, odstranění rekurze • koncová rekurze (tail recursion) • rekurzivní volání je poslední příkaz funkce • lze vesměs přímočaře nahradit cyklem • „plná" rekurze • „zanořující se" volání • např. stromeček o lze přepsat bez použití rekurze za použití zásobníku • rekurzivní podoba často výrazně elegantnější 64/72 Hanojské věže aneb 0 konci světa a klášter kdesi vysoko v horách u města Hanoj • velká místnost se třemi vyznačenými místy • 64 různě velkých zlatých disků • podle věštby mají mniši přesouvat disky z prvního na třetí místo • a až to dokončí ... •<□► 4 ^ t < -ž ► < -š ► 65/72 Hanojské věže: pravidla • N disků různých velikostí naskládaných na sobě • vždy může být jen menší disk položen na větším • možnost přesunout jeden horní disk na jiný kolíček • cíl: přesunout vše z prvního na třetí 66/72 Hanojské věže: řešení JZEZL : [ i i 1 cE i i i c I I 67/72 Hanojské věže: výstup programu »> solve(3, "A", "B", "C") A -> B A -> C B -> C A -> B C -> A C -> B A -> B Hanojské věže: rekurzivní řešení def solve(n, start, end, auxiliary): if n == 1: print(start, "->", end) else: solve(n-1, start, auxiliary, end) solved, start, end, auxiliary) solve(n-1, auxiliary, end, start) 69/72 Hanojské věže: souvislosti • Sierpiňského fraktál • Pascalův trojúhelník https://www.youtube.com/watch?v=8yaoED8j c8Y •<□► 4 ^ t < -ž ► < -E ► 70/72 • Předpokládejme, že y je seznam. Jaký je rozdíl mezi x = yax = y [: ] ? • Jaký je význam pojmů „globální proměnná" a „lokální proměnná "? • Jaké jsou nevýhody použití globálních proměnných? • Jaký je význam pojmů „hluboká kopie" a „mělká kopie"? • Jaké jsou v Pythonu hlavní vestavěné typy? Které z nich jsou měnitelné a které neměnitelné? • Co je to rekurze? • Jak zapíšeme výpočet faktoriálu: a) za použití rekurze, b) bez použití rekurze? 71/72 • představa o reprezentaci v paměti je potřeba • globální, lokální proměnné • předávání parametrů funkcím • mělká vs. hluboká kopie 9 typy, měnitelné, neměnitelné • rekurze příště: regulární výrazy, zpracování textu 72/72