Portál AbcLinuxu, 5. května 2025 23:29

Dotaz: Jak psát správné podmínky

1.11.2017 00:13 Marky
Jak psát správné podmínky
Přečteno: 969×
Odpovědět | Admin
Jak správně v bash psát jednoduché i složitější podmínky? Je z toho něco správně? Díky.
if [ a + [ b - 1 ] < c ]; then
if [[ a + [[ b - 1 ]] < c ]]; then
if (( a + (( b - 1 )) < c )); then
if (( a + $( b - 1 ) < c )); then

Řešení dotazu:


Nástroje: Začni sledovat (0) ?Zašle upozornění na váš email při vložení nového komentáře.

Odpovědi

1.11.2017 00:34 Kit | skóre: 45 | Brno
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Odpovědět | | Sbalit | Link | Blokovat | Admin
Do takových kombinací bych se raději nepouštěl, už kvůli čitelnosti skriptu. Vždycky je lepší to rozepsat do více příkazů. Pokud by to bylo pomalé, je na místě zvolit jiný jazyk.
Komentáře označují místa, kde programátor udělal chybu nebo něco nedodělal.
1.11.2017 10:16 Marky
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
..je na místě zvolit jiný jazyk.
Díky, jiný jazyk? On existuje i jiný jazyk pro práci s linuxem?
1.11.2017 10:30 Kit | skóre: 45 | Brno
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Jistě. Nevím sice, co přesně řešíš, ale pro práci s čísly i stringy je mnohem výhodnější Python. Ze starších nástrojů bys mohl využít Perl, AWK, sed a mnoho dalších. Bývají už nainstalovány.

Shell by se měl používat spíš jako spouštěč jiných programů a výpočty už nechávat na nich.
Komentáře označují místa, kde programátor udělal chybu nebo něco nedodělal.
1.11.2017 13:35 dustin | skóre: 63 | blog: dustin
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Souhlas s pythonem, vhodný jazyk pro skriptování. V bashi jsou složitější věci na palici. Samozřejmě jde všechno, ale blbě se to ladí, je to vše přes ruku. Python si případně odladíš v IDE, pokud to bude něco složitějšího a budeš mít možnost IDE spustit. U složitější věci (více funkcí, práce s řetězci/poli/atd.) ušetříš spoustu času oproti bashi.
1.11.2017 16:59 Marky
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Jejda Python trošičku znám, ale myslel jsem, že je pro Windows. Teď jsem zkoušel nainstalovat na linuxu poslední ver. pythonu ale nainstalovat stejně nejde.
apt install python3.5
Reading package lists... Done
Building dependency tree
Reading state information... Done
python3.5 is already the newest version (3.5.3-1).
0 upgraded, 0 newly installed, 0 to remove and 3 not upgraded.

python -V
Python 2.7.13
Díval jsem se do /etc a jsou tam složky
python
python2.7
python3
python3.5
ale všechny jsou prázdné (v každé je jeden soubor). Kde ty soubory jsou? Díky
1.11.2017 17:12 dustin | skóre: 63 | blog: dustin
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Proč by nešel nainstalovat, když ti to píše

python3.5 is already the newest version (3.5.3-1).

Již jej máš instalovaný, python je na většině běžných distribucí v základu.

V /etc jsou system-wide konfigurace, vlastní software je rozmístěný v odpovídajících adresářích.

Opravdu je potřeba si nejdříve o linuxu něco trošku přečíst.

1.11.2017 18:17 Kit | skóre: 45 | Brno
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
python3 -V
Komentáře označují místa, kde programátor udělal chybu nebo něco nedodělal.
2.11.2017 18:13 Andrej | skóre: 51 | blog: Republic of Mordor
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Jejda Python trošičku znám, ale myslel jsem, že je pro Windows.

Podivná pověra. Žádný kloudný programovací jazyk není "pro Windows" a nevznikl na Windows. (Příznivci C# teď nabíjejí kulomety, ale co už.) Moderní programování jako takové vzniklo na systémech UNIXového typu (s jazyky jako C, C++ nebo Java). Windows jsou prostředí nepřátelské k programování i ke studiu informačních technologií obecně.

ale všechny jsou prázdné (v každé je jeden soubor). Kde ty soubory jsou? Díky

Bohužel neuvádíš, o jakou distribuci se jedná. Předpokládám, že o nějakou založenou na dpkg a spol. Takže:

which python
whereis python
dpkg -l | awk '$2 ~ /python/ {print}'
dpkg -L python
dpkg -L python-minimal

Detaily, co ty příkazy vypisují, jsou v manuálových stránkách.

2.11.2017 17:47 Andrej | skóre: 51 | blog: Republic of Mordor
Rozbalit Rozbalit vše Re: Jak psát správné podmínky

Právě to rozepsání do více příkazů způsobí zoufalou nečitelnost skriptu.

Rozhodně je na místě přečíst si pořádně manuálovou stránku Bashe. Je sice dlouhá a spletitá, ale v drtivé většině skriptů, které vidím, se setkávám s naprostým nepochopením, jak fungují v Bashi datové typy, substituce, pole, asociativní pole, aritmetika a cykly. Někteří autoři skriptů na čtení manuálové stránky rezignovali a kvůli zdánlivě netriviálním trivialitám spouštějí (klidně v cyklech) procesy jako awk nebo sed. Pak jsou skripty pomalé. Manipulace s proměnnými v čistém Bashi, při které se nespouští externí procesy, většinou na zpracování textových dat do stovek MB velikosti zcela postačuje, pokud jde o "efektivitu".

Silnou stránkou Bashe je například jednoduchý multiprocesing, který se dá ve spoustě případů použít pro paralelní zpracování dat, třeba nad výstupem z příkazu find. Přesně tam se podmínky a cykly hodí víc než dobře:

set -e

declare -ri TASKS_TO_RUN=128
declare -ri TASKS_LIMIT=16
declare -i tasks=0

run() {
  local -r task_name="$1"; shift
  if ((tasks > TASKS_LIMIT)); then wait -n; else ((++tasks)); fi
  ( "$@"; echo "${task_name} done."; )&
}

wait_for_completion() { while ((tasks--)); do wait -n; done; }

for ((i = 0; i < TASKS_TO_RUN; ++i)); do
  run "Task ${i}" sleep ".$((100 * RANDOM / 32768))"
done
wait_for_completion
2.11.2017 18:55 Kit | skóre: 45 | Brno
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Proč tam máš funkci wait_for_completion, když by měl stačit vestavěný příkaz wait?

Používání globálních proměnných je fakt hnus.
Komentáře označují místa, kde programátor udělal chybu nebo něco nedodělal.
2.11.2017 21:06 Andrej | skóre: 51 | blog: Republic of Mordor
Rozbalit Rozbalit vše Re: Jak psát správné podmínky

Jsem jedno ucho, jak se v Bashi v tomhle konkrétním případě zbavíš globálních proměnných. Tak prosím, příklady jsou vítané. A pokud možno aby nebyly hnus.

Mimochodem, globální konstanty rozhodně nejsou hnus a globální počítadla taky ne — to jenom připomínám pro úplnost. :-)

Asi si umíš představit, že wait_for_completion by mohl mít (v původní, složitější verzi takového skriptu) za úkol třeba vypisovat údaje o počtu zbývajících procesů, že ano. :-)

2.11.2017 22:44 Kit | skóre: 45 | Brno
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Dobrá, tohle se ti určitě nebude líbit, ale je mi to bližší. Když si místo toho cyklu dosadíš třeba find, zbavíš se konstanty TASKS_TO_RUN a budeš to mít v podobě, kterou používám:
#!/bin/bash
set -e

declare -ri TASKS_TO_RUN=12
declare -ri TASKS_LIMIT=4

for ((i = 0; i < TASKS_TO_RUN; ++i)); do
    echo -ne "./subproces.sh Task_${i} sleep .$((100 * RANDOM / 32768)) \00"
done | xargs -0 -n 1 -P $TASKS_LIMIT bash -c
Soubor subproces.sh zpravidla není skript, ale konkrétní program, který data ze souboru zpracuje. Tohle je jen pro ilustraci:
#!/bin/bash
set -e

declare -r task_name="$1"
shift
"$@"
echo "${task_name} done."
Jak vidíš, žádnou globální proměnnou jsem nepoužil. Data předávám pouze přes roury a parametry.
Komentáře označují místa, kde programátor udělal chybu nebo něco nedodělal.
3.11.2017 13:17 Andrej | skóre: 51 | blog: Republic of Mordor
Rozbalit Rozbalit vše Re: Jak psát správné podmínky

To nakonec skvěle ilustruje, do jaké míry je ten „hnus“ spíš věcí názoru. :-) Řešení s xargs je svým způsobem pěkné, ale nehodí se třeba pro (zmíněné) hlášení počtu zbývajících procesů nebo pro (méně triviální) přesměrování standardního vstupu i výstupu z těch procesů, což můžou být pojmenované roury (mkfifo), soubory atd. Samozřejmě to taky jde — s použitím toho odděleného skriptu, který to zařídí —, ale pak se nabízí znova otázka, co je „hnus“ a co až tolik ne.

Problém s globální proměnnou je vlastně spíš slovíčkaření, protože přesměrování celého výstupu do xargs je v mnoha směrech horší než globální proměnná. Přístup s run a wait_for_completion umožňuje skriptu normálně používat stdout i stderr, paralelně dělat i jiné úkony než generování argumentů pro xargs (ptát se uživatele v terminálu, jestli chce spustit i tyhle další procesy nebo třeba ještě ne), tu a tam spustit něco na pozadí pomocí run a pak na to třeba počkat, když je nutná synchronizace mezi nějakými fázemi toho výpočtu.

Řešení s xargs bude vždycky tomu shellu blokovat některý ze standardních výstupů. Může sice číst z pojmenované roury, ale do té roury se pak musí přesměrovat celý blok (for, while apod.), protože přesměrovat tam třeba něco jen tak z echo znamená, že se ta roura hned zavře a xargs už ji dál číst nebude. To není moc flexibilní. Můžu jistě celý vstup pro xargs vygenerovat předem do pole řádků nebo jiné struktury, ale to jsme zase u toho hnusu.

Mimochodem, právě různá řešení s xargs jsem ve spoustě skriptů nahradil jednoduchými subshelly, protože xargs není dost flexibilní pro účely, ke kterým ty skripty byly.

Další killer feature ampersandu je, že po každém & se dá přečíst $!. Něco takového xargs taky neumí. Tedy lze mít například pole s PID všech spuštěných procesů na pozadí a lze čekat (pomocí wait) na každý z nich zvlášť a získat a zpracovat jeho návratovou hodnotu.

3.11.2017 13:32 Andrej | skóre: 51 | blog: Republic of Mordor
Rozbalit Rozbalit vše Re: Jak psát správné podmínky

Pro ilustraci těch výsledků procesů na pozadí:

background_pids=()
exit_codes=()
# ...
run_something &
background_pids[${#background_pids[@]}]="$!"
run_something_else &
background_pids[${#background_pids[@]}]="$!"
# ...
for idx in "${!background_pids[@]}"; do
  wait "${background_pids[idx]}"
  exit_codes[idx]="$?"
done
# ...

Tohle^^^ xargs sice možná dá taky, ale způsobem příliš obskurním.

1.11.2017 01:35 .
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Odpovědět | | Sbalit | Link | Blokovat | Admin
[ a [[ neumí aritmetické operace a na porovnání menší / větší je -lt / -gt.
[[ umí i < / >, ale to porovnává řetězce (tj. 10 je menší než 5).
$() zachytí výstup programu. Možná jde o překlep a myslel jsi $(()).

Na aritmetické vyhodnocení je (()) a $(()). Třetí možnost je správně (a ty vnitřní závorky není potřeba zdvojovat), a pokud jde o překlep, tak i ta čtvrtá.
1.11.2017 10:13 Marky
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Díky, tak že takto
if (( a + ( b - 1 ) < c )); then
nebo takto
if (( a + $(( b - 1 )) < c )); then
ale když chci výsledek do proměnné tak takto?
VAR=$( a + b )
1.11.2017 10:46 .
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
VAR=$(( a + b ))
1.11.2017 16:44 Marky
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Hm a kdy se teda používá tohle?
$( a + b )
1.11.2017 18:20 NN
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
$() zachytí výstup programu.
foo=$(echo "bar")
2.11.2017 17:01 Andrej | skóre: 51 | blog: Republic of Mordor
Rozbalit Rozbalit vše Re: Jak psát správné podmínky
Odpovědět | | Sbalit | Link | Blokovat | Admin

Tohle přece můžeš snadno zkusit. První, druhý a čtvrtý příklad hodí syntaktickou chybu, takže to asi nebude ono.

Třetí příklad je v podstatě OK, jen není důvod mít tam dvojité závorky, když už jsi v aritmetickém výrazu.

a=3
b=4
c=6
if (( a + (b - 1) < c )); then echo jo; else echo ne; fi

Bash umí taky (trochu) specifikovat datové typy (integer, indexované pole, asociativní pole) a typ integer umí jednodušší syntaxi přiřazení (bez $(())), jako například proměnná e2 níže.

declare -i e2
e1='a + (b - 1)'
e2='a + (b - 1)'
e3=$((a + (b - 1)))

echo "$e1"  # a + (b - 1)
echo "$e2"  # 6
echo "$e3"  # 6

Podmínky a výrazy se samozřejmě dají použít taky ve for-cyklu.

c=20
for ((a = 1, b = 10; a + (b - 1) < c; a += 2, ++b)); do
  echo "a: ${a}  b: ${b}  c: ${c}  a + (b - 1): $((a + (b - 1)))"
done

Nebo v jiných cyklech.

a=1
b=10
c=20
while ((a + (b - 1) < c)); do
  echo "a: ${a}  b: ${b}  c: ${c}  a + (b - 1): $((a + (b - 1)))"
  ((a += 2))
  ((++b))
done

Založit nové vláknoNahoru

Tiskni Sdílej: Linkuj Jaggni to Vybrali.sme.sk Google Del.icio.us Facebook

ISSN 1214-1267, (c) 1999-2007 Stickfish s.r.o.