Živá analýza proměnných - Live variable analysis
v překladače, živá analýza proměnných (nebo jednoduše analýza živosti) je klasika analýza toku dat vypočítat proměnné to jsou žít v každém bodě programu. Proměnná je žít v určitém okamžiku, pokud obsahuje hodnotu, která může být v budoucnu potřebná, nebo ekvivalentně, pokud lze její hodnotu přečíst před příštím zápisem proměnné.
Příklad
Zvažte následující program:
b = 3c = 5a = f (b * c)
Sada živých proměnných mezi řádky 2 a 3 je {b
, C
} protože oba jsou použity v násobení na řádku 3. Ale sada živých proměnných po řádku 1 je pouze {b
}, protože proměnná C
je aktualizován později, na řádku 2. Hodnota proměnné A
se v tomto kódu nepoužívá.
Všimněte si, že úkol A
mohou být odstraněny jako A
se později nepoužije, ale není k dispozici dostatek informací k odstranění celého řádku 3 jako F
může mít vedlejší účinky (tisk před naším letopočtem
možná).
Vyjádření z hlediska rovnic toku dat
Analýza živosti je analýza „zpětně může“. Analýza se provádí v a zpět pořadí a tok dat operátor soutoku je nastavit unii. Jinými slovy, pokud použijeme analýzu živosti na funkci s konkrétním počtem logických větví, provede se analýza od konce funkce, která pracuje od začátku (tedy „zpět“), a proměnná se považuje za živou, pokud kterákoli z větví pohybujících se vpřed ve funkci může potenciálně (tedy „může“) potřebovat aktuální hodnotu proměnné. To je v rozporu s analýzou „zpětně nutné“, která by místo toho vynucovala tuto podmínku u všech větví pohybujících se vpřed.
Rovnice toku dat použité pro daný základní blok s a opuštění bloku F v živé analýze proměnných jsou následující:
- : Sada proměnných, které se používají v s před jakýmkoli přiřazením.
- : Sada proměnných, kterým je přiřazena hodnota v s (v mnoha knihách[je zapotřebí objasnění ], KILL (s) je také definován jako sada proměnných, kterým je přiřazena hodnota v s před jakýmkoli použitím, ale to nemění řešení rovnice toku dat):
Stav bloku je sada proměnných, které jsou aktivní na začátku bloku. Jeho out-state je sada proměnných, které jsou aktivní na jeho konci. Out-state je svazek in-stavů nástupců bloku. Přenosová funkce příkazu se aplikuje tak, že se proměnné, které jsou zapsány, stanou mrtvými, pak se proměnné, které se čtou, stanou živými.
Druhý příklad
// in: {} b1: a = 3; b = 5; d = 4; x = 100; // x se nikdy nepoužívá později, tedy není v out set {a, b, d} if a> b then // out: {a, b, d} // sjednocení všech (ne) nástupců b1 => b2: {a, b} a b3: {b, d} // in: {a, b} b2: c = a + b; d = 2; // out: {b, d} // in: {b, d} b3: endif c = 4; návrat b * d + c; // out: {} |
Stav b3 obsahuje pouze b a d, od té doby C bylo napsáno. Out-state of b1 is the union of the in-states of b2 and b3. Definice C v b2 lze odstranit, protože C není aktivní okamžitě po prohlášení.
Řešení rovnic toku dat začíná inicializací všech stavů a stavů do prázdné sady. Pracovní seznam se inicializuje vložením výstupního bodu (b3) do pracovního seznamu (typický pro zpětný tok). Jeho vypočítaný stav se liší od předchozího, takže jsou vloženi jeho předchůdci b1 a b2 a proces pokračuje. Pokrok je shrnut v tabulce níže.
zpracovává se | out-state | starý stát | nový stát | pracovní seznam |
---|---|---|---|---|
b3 | {} | {} | {b, d} | (b1, b2) |
b1 | {b, d} | {} | {} | (b2) |
b2 | {b, d} | {} | {a, b} | (b1) |
b1 | {a, b, d} | {} | {} | () |
Všimněte si, že b1 byl zadán do seznamu před b2, což vynutilo zpracování b1 dvakrát (b1 bylo znovu zadáno jako předchůdce b2). Vložení b2 před b1 by umožnilo dřívější dokončení.
Inicializace s prázdnou sadou je optimistická inicializace: všechny proměnné začínají jako mrtvé. Všimněte si, že out-stavy se nemohou zmenšit z jedné iterace na další, i když out-state může být menší než in-state. To je patrné ze skutečnosti, že po první iteraci se může stav změnit pouze změnou stavu. Vzhledem k tomu, že stav začíná jako prázdná sada, může růst pouze v dalších iteracích.