Portál AbcLinuxu, 17. května 2024 22:04

To jsem blázen, aneb co bash dělá s tou proměnnou?

4.5.2009 16:03 | Přečteno: 1358× | Za vším hledej Linux

Chtěl jsem si ušetřit práci zpracováním seznamu předaného smyčce a narazil na zajímavý problém. Nějak prapodivně mi funguje echo ve smyčce.

Asi nejlépe to bude demonstrovat na příkladu:

user@stroj:~$ cat osoby.txt 
ja
ty
on
ona
ono
user@stroj:~$ for i in $(<osoby.txt); do echo "aa$i" ; done
aaja
aaty
aaon
aaona
aaono
user@stroj:~$ for i in $(<osoby.txt); do echo "$iaa" ; done





user@stroj:~$ for i in $(<osoby.txt); do echo "$i aa" ; done
aa
aa
aa
aaa
aao
Nějak se nemůžu zbavit dojmu, že by správně měl výsledek druhé smyčky vypadat takto:
user@stroj:~$ for i in $(<osoby.txt); do echo "$iaa" ; done
jaaa
tyaa
onaa
onaaa
onoaa
user@stroj:~$ for i in $(<osoby.txt); do echo "$i aa" ; done
ja aa
ty aa
on aa
ona aaa
ono aa
Protože když udělám spojení proměnné s řetězcem přímo, tak to vypadá takto..
user@stroj:~$ osoba="nekdo"
user@stroj:~$ echo "$osoba jiny"
nekdo jiny
Je to normální? Nebo dělám někde něco blbě? Nebo mi už hrabe?        

Hodnocení: 50 %

        špatnédobré        

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

Komentáře

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

Vložit další komentář

4.5.2009 16:13 Ondrej 'SanTiago' Zajicek
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Odpovědět | Sbalit | Link | Blokovat | Admin

"$iaa" je evidentne blbe - promenou iaa tam nemas. Kdyztak "${i}aa". Nevim, proc ti nefunguje "$i aa", to by melo byt v poradku. Mozna zkus nahradit vyraz  v for cyklu na `cat soubor` (a nebo jeste lepe pouzit 'while read i'), zda neni problem v tom.

4.5.2009 16:18 Aleš Kapica | skóre: 51 | blog: kenyho_stesky | Ostrava
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Dík z upozornění - už jsem byl z toho tak zpitomělý že mě to nenapadlo. I když se nemůžu ubránit dojmu že pokud odpovídá část za $ názvu proměnné, tak by měl shell provést substituci. Mám tedy pocit, že alespoň dříve to tak fungovalo.
4.5.2009 16:26 Michal Kubeček | skóre: 72 | Luštěnice
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Nevim, proc ti nefunguje "$i aa", to by melo byt v poradku.

To je opravdu v pořádku:

mike@unicorn:~/tmp> for i in $(<osoby.txt); do echo "$i aa" ; done
ja aa
ty aa
on aa
ona aa
ono aa

Problém tazatele bude nejspíš v něčem jiném, co v této ukázce není vidět.

4.5.2009 16:31 Michal Kubeček | skóre: 72 | Luštěnice
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Už tuším, kde by mohl být problém. Poslední příklad tazatele odpovídá logickému chování v případě, že by ta mezera v "$i aa" byl ve skutečnosti CR.
4.5.2009 16:39 Aleš Kapica | skóre: 51 | blog: kenyho_stesky | Ostrava
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Je to tak. Napadlo nás to zhruba ve stejnou chvilku.
krab1k avatar 4.5.2009 16:15 krab1k | skóre: 11 | Brno
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Odpovědět | Sbalit | Link | Blokovat | Admin
Tak minimálně echo "$iaa" by mělo být takto: echo "${i}aa", ne?
4.5.2009 16:20 Aleš Kapica | skóre: 51 | blog: kenyho_stesky | Ostrava
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Výsledek je stejný jako v případě když tam je mezera. Z mně záhadného důvodu dojde k substituci.
4.5.2009 16:29 Michal Kubeček | skóre: 72 | Luštěnice
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?

Proč záhadného? Záhadné by bylo, kdyby k ní nedošlo. Vyskytne-li se za dolarem písmeno* nebo podtržítko, pak se všechno, co následuje až k prvnímu znaku, který není písmeno*, číslice nebo podtržítko, považuje za název proměnné a výsledkem expanze je hodnota této proměnné. Neexistuje-li, výsledkem expanze je prázdný řetězec.


* anglické abecedy

4.5.2009 16:38 Aleš Kapica | skóre: 51 | blog: kenyho_stesky | Ostrava
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Ano, ovšem k substituci dojde i v případě druhém. A já už taky vím proč. ;-). Ten soubor, který zpracovávám mi totiž poslal kolega. Až teprve poté co jsem si odzkoušel tu smyčku na jiném stroji, mě napadlo co za tím vězí. Chvilka napětí..

Kouknul jsem se na ně hexaeditorem.. A byly tam! Ty ZPROPADENÉ widlácké konce řádků! Ano přátelé. Za nefunkčností skriptu stála kombinace "0D 0A" na konci každého řádku.
4.5.2009 16:40 CEST
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Hehe:) Jo, svet Windows dokaze clovek privest k silenstvi:) BTW: Byli jsme na stejne:)
4.5.2009 16:45 Ondrej 'SanTiago' Zajicek
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?

No v tom pripade skript vlastne perfektne fungoval.

4.5.2009 16:19 čavo
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Odpovědět | Sbalit | Link | Blokovat | Admin
Premennú treba ohraničiť aby ju správne našiel: namiesto "$iaa" "${i}aa"
4.5.2009 16:38 CEST
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Odpovědět | Sbalit | Link | Blokovat | Admin
OK, zapis $iaa je zacatecnicka chyba:) Tohle plati u vsech jazyku, ze za identifikatorem promenne je nazev promenne vsechno, co odpovida znakum, ktere se muzou vyskytovat v nazvu promenne - je to tak v bashi, perlu, php a zrejme i dalsich. Jak bys totiz potom chtel rozlisit zapis $iaa, jestli je to promenna $i a text aa, nebo promenna $ia a text "a" nebo promenna "$iaa". Proto se pouzivaji zapisy "${promenna}text", kterym se jasne vymezi, co je promenna.

A ten problem s tim "$i aa" je takovej docela chytacek:) uz jsem se s tim taky setkal:) Je to problem mezi svetem Windows a Unix. Tipuju, ze jsi ten soubor vytvoril na woknech a pak jsi to bez konverze zkousel na linuxu. Cili radky obsahuji:
ja[CR][LF]
ty[CR][LF]
on[CR][LF]
ona[CR][LF]
ono[CR][LF]
Bash potom do promenny $i nacpe napr. "ja[CR]" a [LF] odmaze a kdyz to pak tisknes, tak se vytiskne $i jako "ja[CR]" a [CR] je carriage return (posun kurzoru na zacatek radky), takze zbytek textu " aa" se napise zase od zacatku radky, proto prepise ten text z promenny $i:) Hezkej chytak, co?:)
4.5.2009 16:46 Aleš Kapica | skóre: 51 | blog: kenyho_stesky | Ostrava
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?
Jak už jsem napsal o kousek výše - soubor jsem nevytvářel já. Jinak zápis $iaa už byl pouze projevem čirého zoufalství, kdy už člověk zkouší kdejakou kravinu.

Už jsem se s tím také pochopitelně setkal, jen to už je dost dávno a navíc - byl to problém primárně související s mixováním konců řádků v php souborech. Každopádně dík za vysvětlení procesu jak dochází k té substituci.

Svůj díl na intelektuálním výkonu jistě sehrálo i to, že ledva jsem na problém narazil, přišel kolega s tím, že z databáze zmizely úkoly od studentů, takže jsem ještě mezi tím řešil havarované tabulky v databázi.
17.5.2009 00:22 Pavel
Rozbalit Rozbalit vše Re: To jsem blázen, aneb co bash dělá s tou proměnnou?

Jen tak pro zajímavost,  to tvrzení "Tohle plati u vsech jazyku" je přecejenom trochu moc silné.

Existují i jazyky a skripty, kde to funguje jinak. Oba níže uvedené příkazy pro vypsání hodnoty vypíší shodný řetězec. Jaký GNU program čte tuto syntaxi ponechávám na laskavém čtenáři.

a=testa
$(warning testa="$atest")
$(warning testa="$(a)test")

Založit nové vláknoNahoru

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