Portál AbcLinuxu, 3. prosince 2025 06:41
smyslem char** je změnit někde adresu ukazatele
Pokud chceme, aby se změna obsahu ukzatele (tedy kam ukazuje) projevila i vně funkce (tedy jakoby se předávalo odkazem), tak pracujeme s "**ret" (v hlavičce funkce). Díky!
typ char ** char * char adresa: 2008 2004 2000 data: 2004 -> 2000 -> 'A'Rozdíl mezi těma funkcema ve volání může být třeba takový:
char c = 'A'; char * cp = &c; /*cp == 2000*/ char ** cpp = &cp; /* cpp == 2004*/ /*volani se stejnym vysledkem*/ funkce1(&c); funkce1(cp); /*volani se stejnym vysledkem*/ funkce2(&cp); funkce2(cpp);pokud navíc:
int funkce2(char **ret)
{
return funkce1(*ret);
}
Tak se to všechno bude chovat stejně.
Ty Cckove komentare tomu zrovna nepomahaji :)char c = 'A'; char * cp = &c; /*cp == 2000*/
char c = 'A'; char * cp = &c; // cp == 2000 char ** cpp = &cp; // cpp == 2004
Děkuju moc. Už to chápu.
Nějak nechápu, co na tom chceš ještě vysvětlit. Připadá mi, že v tom hledáš nějakou magii, která v tom není.
První funkce přebírá ukazatel na znak, druhá funkce přebírá ukazatel na ukazatel na znak. Možná ti pomůže vysvětlení na anglické Wikipedii. A nebo si to zkus načrtnout na papír.
Ukazatele se dají obecně indexovat jako pole (dokonce i do mínusu), takže *ret ve funkci1 je skutečně to samé jako ret[0] v téže funkci (jak se tu řešilo nedávno, ret[0] a 0[ret], je defakto *((ret) + (0)), což je zas ekvivalentní k *ret). Rozdíl mezi ukazatelem a polem se ukáže v momentě, kdy se pokusíš zjistit jejich velikost.
A teď jsem tě možná pomátl dočista. :)
a[1000] -> 'A','B','C','D'...'\0'Druha funkce2 bere ukazatel na ukazatele na chary, cili od adresy ret mas seznam cisel, ktere udavaji adresy, kde zacinaji jednotlive chary. Tady je ale char **a a[3000] -> 1000,1200,1400,1600
1000 -> 'a','a','a','a',...'\0' 1200 -> 'e','e','e','e',...'\0' 1400 -> 'c','c','c','c',...'\0'nebo se pletu?
int funkce1(char *ret)
{
/* Můžu si tedy dovolit toto: */
*ret = 'A';
}
ne tak uplne. treba volani funkce1("Foobar"); ti neprojde, protoze retezec je s nejvetsi pravdepodobnosti v casti pameti, ktera se neda menit.
"Foobar" je, afaik, const char*.C prekladac to dovoli, protoze se nemeni ukazatel ale pamet na kterou ukazuje. C++ to nedovoli.Takhle použitý const právě naopak znamená, že se *nesmí měnit obsah té paměti. * Mnohé překladače pouze varují.
A přidáme si do toho makra, která sice mohou být úžasná, ale zároveň i megaultrahumusná a vznikne dobrý chaos... Například práce s ASN.1 v OpenSSL je naprosto "úžasná". Zavoláte jedno makro (od kterého není pořádný popis v dokumentaci) a ono to udělá asi milión volání funkcí, přetypovávání a já nevím čeho všeho. Definice makra není pořádně k dispozici, protože je nějaká hromadná, či jak se to jmenuje atd...
Nu, to jsem se ale nechal trochu unést, v principu jsem chtěl říct jen to, že tahle práce s pointery je v C alfou a omegou všeho a je potřeba jí perfektně ovládát... Až pak budete luštit/tvořit konstrukce typu (sc_path_t *) &(env->file_ref), budete to sakra potřebovat...

)
Děkuju za názor.
Tiskni
Sdílej:
ISSN 1214-1267, (c) 1999-2007 Stickfish s.r.o.