TITO. ltö. C-ohjelmointi, syksy 2006: Funktiot. Aliohjelmaesimerkki. Esimerkki: potenssiin korotus. Luento 4, ti

Samankaltaiset tiedostot
C-ohjelmointi: Funktiot. Luento: viikko 2

C-ohjelmointi, syksy 2008: Funktiot

TITO. C-ohjelmointi, syksy 2008: Funktiot. ltö. Aliohjelmaesimerkki. Esimerkki: potenssiin korotus. Luento 4, pe

C-ohjelmointi, kevät t 2006: Funktiot. Luento 4, ti Tiina Niklander

C-ohjelmointi, syksy 2006: Funktiot

TITO. Sisältö. C-ohjelmointi, kevät 2006: Funktiot. Aliohjelmaesimerkki. Yleistä. Parametrit. Esimerkki: potenssiin korotus

C-ohjelmointi, kevät 2006: Funktiot

Rakenne. C-ohjelmointi: Tietorakenteita ja tyyppejä. Tyyppimuunnoksia aritmeettisten tyyppien välillä. Tyyppimuunnokset. & (bitti and) Bittioperaatiot

Jakso 4 Aliohjelmien toteutus

C-ohjelmointi: Osoittimet

Luento 4 Aliohjelmien toteutus

Jakso 4 Aliohjelmien toteutus

Aliohjelmatyypit (2) Jakso 4 Aliohjelmien toteutus

Luento 4 Aliohjelmien toteutus

Luento 4 Aliohjelmien toteutus

Luento 4 Aliohjelmien toteutus. Tyypit Parametrit Aktivointitietue (AT) AT-pino Rekursio

Jakso 4 Aliohjelmien toteutus. Tyypit Parametrit Aktivointitietue (AT) AT-pino Rekursio

Jakso 4 Aliohjelmien toteutus

Luento 4 (verkkoluento 4) Aliohjelmien toteutus

Luento 4 (verkkoluento 4) Aliohjelmien toteutus

Aliohjelmatyypit (2) Jakso 4 Aliohjelmien toteutus

Sisältö. C-ohjelmointi Luento 5: Osoittimet. Keko (heap) Pino (stack) Muistinhallinta Java vs C. Prosessin rakenne

C-ohjelmointi Luento 5: Osoittimet

C-ohjelmointi Luento 5: Osoittimet Tiina Niklander

Lyhyt kertaus osoittimista

Monipuolinen esimerkki

Moduli 5: Kehittyneitä piirteitä

815338A Ohjelmointikielten periaatteet Harjoitus 5 Vastaukset

C-kielessä taulukko on joukko peräkkäisiä muistipaikkoja, jotka kaikki pystyvät tallettamaan samaa tyyppiä olevaa tietoa.

Tieto ja sen osoite (3) Jakso 3 Konekielinen ohjelmointi (TTK-91, KOKSI) Osoitinmuuttujat. Tieto ja sen osoite (5)

Loppukurssin järjestelyt C:n edistyneet piirteet

Loppukurssin järjestelyt

Tietueet. Tietueiden määrittely

TIETORAKENTEET JA ALGORITMIT

C-kurssi syksy ltö. Luennon sisält. Luento 2: tyypit, rakenteet, makrot Tyypit. signed, unsigned short,, long Vakiot const Rakenteet

Luennon sisält. ltö. C-kurssi syksy ääreet: int ja char. Yksinkertaiset tyypit. Kokonaisluvut. Merkit

Operaattoreiden ylikuormitus. Operaattoreiden kuormitus. Operaattoreiden kuormitus. Operaattoreista. Kuormituksesta

C-kurssi kevät t Luento 2: tyypit, rakenteet, makrot

Jakso 3 Konekielinen ohjelmointi (TTK-91, KOKSI)

Sisällys. 1. Omat operaatiot. Yleistä operaatioista. Yleistä operaatioista

Jakso 3 Konekielinen ohjelmointi (TTK-91, KOKSI)

Luennon sisältö. Taulukot (arrays) (Müldnerin kirjan luku 10) Yksiulotteiset taulukot. Mikä taulukko on? Taulukko-osoitin. tavallinen osoitin

1. Omat operaatiot 1.1

C-ohjelmointi, syksy 2006

C-ohjelmointi, syksy Yksiulotteiset taulukot Moniulotteiset taulukot Dynaamiset taulukot. Binääritiedostot. Luento

Tietorakenteet ja algoritmit

1.3Lohkorakenne muodostetaan käyttämällä a) puolipistettä b) aaltosulkeita c) BEGIN ja END lausekkeita d) sisennystä

Osoitin ja viittaus C++:ssa

Muuttujien roolit Kiintoarvo cin >> r;

C-kurssi kevät Luennon sisältö

Luennon sisältö. C-kurssi kevät Tasokokeen kohta 1: Taulukon järjestäminen. Tasokokeen kohta 2. Tasokokeen kohta 2. Tasokokeen kohta 3

815338A Ohjelmointikielten periaatteet Harjoitus 2 vastaukset

Rakenteiset tietotyypit Moniulotteiset taulukot

Ohjelmassa muuttujalla on nimi ja arvo. Kääntäjä ja linkkeri varaavat muistilohkon, jonne muuttujan arvo talletetaan.

Harjoitustyö: virtuaalikone

1.1 Pino (stack) Koodiluonnos. Graafinen esitys ...

Luennon sisältö Tyypit int, char, float, double signed, unsigned short, long Vakiot const Rakenteet if, for, while, switch, do-while Syöttö ja tulostu

815338A Ohjelmointikielten periaatteet Harjoitus 3 vastaukset

Luento 3 Konekielinen ohjelmointi (TTK-91, KOKSI)

tietueet eri tyyppisiä tietoja saman muuttujan arvoiksi

Osoittimet ja taulukot

ITKP102 Ohjelmointi 1 (6 op)

Tietorakenteet ja algoritmit

1.3 Lohkorakenne muodostetaan käyttämällä a) puolipistettä b) aaltosulkeita c) BEGIN ja END lausekkeita d) sisennystä

Muistin käyttö. Muistin käyttö. Muistin käyttö. Muistin käyttö. Muistin käyttö. Muistin käyttö. Muistin käyttö C-ohjelmassa

Tietorakenteet ja algoritmit

Jakso 3 Konekielinen ohjelmointi (TTK-91, KOKSI)

ICS-C2000 Tietojenkäsittelyteoria Kevät 2016

LOAD R1, =2 Sijoitetaan rekisteriin R1 arvo 2. LOAD R1, 100

ITKP102 Ohjelmointi 1 (6 op)

#include <stdio.h> // io-toiminnot. //#define KM_MAILISSA int main( ){

15. Ohjelmoinnin tekniikkaa 15.1

Tiedostot. Tiedostot. Tiedostot. Tiedostot. Tiedostot. Tiedostot

Sisällys. 12. Näppäimistöltä lukeminen. Yleistä. Yleistä

Modulaarisessa ohjelmoinnissa jaetaan ohjelma osiin (moduuleihin), jotka ovat yksinkertaisia ja lyhyitä.

Sisältö. 22. Taulukot. Yleistä. Yleistä

Jakso 3 Konekielinen ohjelmointi (TTK-91, KOKSI)

2) Aliohjelma, jonka toiminta perustuu sivuvaikutuksiin: aliohjelma muuttaa parametrejaan tai globaaleja muuttujia, tulostaa jotakin jne.

C-ohjelma. C-ohjelma. C-ohjelma. C-ohjelma. C-ohjelma. C-ohjelma. Operaatioiden suoritusjärjestys

7. Oliot ja viitteet 7.1

15. Ohjelmoinnin tekniikkaa 15.1

Luento 3 Konekielinen ohjelmointi (TTK-91, KOKSI) Muuttujat Tietorakenteet Kontrolli Optimointi Tarkistukset

Luento 3 Konekielinen ohjelmointi (TTK-91, KOKSI)

Osoittimet ja taulukot

Luento 3 Konekielinen ohjelmointi (TTK-91, KOKSI) Muuttujat Tietorakenteet Kontrolli Optimointi Tarkistukset

Tiedon sijainti suoritusaikana. Luento 3 Konekielinen ohjelmointi (TTK-91, KOKSI)

Tiedon sijainti suoritusaikana (3) Luento 3 Konekielinen ohjelmointi (TTK-91, KOKSI) Miten tietoon viitataan? (4)

19. Olio-ohjelmointia Javalla 19.1

Ohjelmoinnin jatkokurssi, kurssikoe

Koottu lause; { ja } -merkkien väliin kirjoitetut lauseet muodostavat lohkon, jonka sisällä lauseet suoritetaan peräkkäin.

Metodit. Metodien määrittely. Metodin parametrit ja paluuarvo. Metodien suorittaminen eli kutsuminen. Metodien kuormittaminen

Luokassa määriteltävät jäsenet ovat pääasiassa tietojäseniä tai aliohjelmajäseniä. Luokan määrittelyyn liittyvät varatut sanat:

Tietorakenteet ja algoritmit

C++11 lambdat: [](){} Matti Rintala

3. Binääripuu, Java-toteutus

Ohjelmoinnin perusteet Y Python

Listarakenne (ArrayList-luokka)

Aliohjelmien toteutus Suoritin ja väylä

Imperatiivisen ohjelmoinnin peruskäsitteet. Meidän käyttämän pseudokielen lauseiden syntaksi

ITKP102 Ohjelmointi 1 (6 op)

Transkriptio:

Sisält ltö C-ohjelmointi, syksy 2006: Funktiot Luento 4, ti 15.9.2006 Yleistä Parametrit Paluuarvo ja sen käyttk yttö Ohjelmointityylistä: : modulaarisuus Funktion esittelyt otsikkotiedostoon Monen ohjelmatiedoston käyttk yttö Näkyvyyssäännöt Osoitintyyppi Viiteparametrit Funktioparametrit Yleistä Funktiot ovat kaikki irrallisia ja samalla tasolla (ei sisäkk kkäisiä funktioita) Ohjelmassa on aina tasan yksi main-funktio Suoritus alkaa tästt stä funktiosta Näkyvyyttä voi rajoittaa vain tiedostoittain ja kirjoitusjärjestyksell rjestyksellä Funktioilla on vain arvoparametreja Viiteparametrit on toteutettava osoitintyypin avulla Funktio voi saada parametrinaan myös s toisen funktion osoitteen Funktioiden syntaksi kuten Javan metodit TITO Kaikki viitteet näihin tehdään suhteessa FP:hen Aliohjelmaesimerkki int fa (int x, y) int z = 5; z = x * z + y; return (z); T = fa (200, R); FP SP paluuarvo paramx paramy vanha PC vanha FP paik. z vanha R1 aliohjelman toteutus: retfa EQU -4 parx pary EQU -3 EQU -2 locz EQU 1 fa ks. fa.k91 PUSH SP, =0 ; alloc Z PUSH SP, R1 ; save R1 LOAD R1,=5; init Z STORE R1, locz (FP) prolog LOAD R1, parx (FP) MUL R1, locz (FP) ADD R1, pary (FP) STORE R1, locz (FP) STORE R1, retfa (FP) POP SP, R1; recover R1 SUB SP, =1 ; free Z EXIT SP, =2 ; 2 param. epilog Esimerkki: potenssiin korotus Parametrit #include <stdio.h> int power (int m, int n); int main( ) for (i=0; i < 20; ++i) printf (%d %d %d \n, i, power(2,i), power(-3,i)); int power (int base, int n) int i, p=1; for (i=1; i<=n; ++i) p = p*base; return p; Funktion esittely Tai: int power (int, int); Oltava ennen käyttk yttöä Funktion kutsu Parametreille arvot Funktion määm äärittely Voi olla esittelyn yhteydessä tai erikseen HUOM: ei puolipistettä otsikossa Aina arvoparametreja (kuten Javan metodeilla) Arvo kopioidaan pinoon Käytetään n funktion sisäll llä parametrin nimellä Funktion sisäll llä parametrit käyttk yttäytyvät t kuin paikalliset muuttujat Tarvittaessa viiteparametreja on välitettv litettävä muuttujan osoite. Erityisellä tyypillä void voi kertoa, jos funktiolla ei ole parametreja 1

void ja tyyppimuunnokset Esittely (declaration): int f() on sama kuin int f(void) Kutsu (call): f(); on sama kuin kuin (void)f(); Parametrien arvot muunnetaan kutsussa tarvittaessa (kunhan tyypit tiedossa!). Vastaava muunnos tehdää ään paluuarvolle. int f(int); double x = f(1.2); Paluuarvo ja sen käyttk yttö Funktion paluuarvo on aina määm ääriteltävä Yksinkertainen tyyppi (ei voi olla rakenne kuten Javassa) Voi olla myös void (eli ei paluuarvoa) Kutsuja pääp äättää mitä paluuarvolla tekee Kutsu lausekkeessa -> > arvo suoraan käyttk yttöön, sijoitetaan muuttujan arvoksi tai jopa jätetj tetään n hyödynt dyntämättä (valitettavasti sallittua!) exit -funktio Ohjelman suorituksen voi missä tahansa ja koska tahansa lopettaa kutsumalla exit- funktiota. exit(int code); double f(double x) if(x < 0) fprintf(stderr,, "negative x in %s\n",% FILE ); exit(exit_failure); /* no return */ return sqrt(x); Paluuarvot: : EXIT_SUCCESS tai EXIT_FAILURE Funktion dokumentointi Esittelyn tai määrittelyn (tai molempien) ) on syytä kuvata funktion käyttäytyminen: Function: nimi Purpose: Yleiskuvaus toiminnasta (tyypillisesti kuvaa, miten funktion oletetaan toimivan) Inputs: Luettelo parametreista ja käytetyistä globaaleista muutt. Returns: Paluuarvo Modifies: Luettelo parametreista ja globaaleista muuttujista joiden arvo muuttuu toiminnan seurauksena (myös: sivuvaikutukset!!) Error checking: Funktion tekemät parametrien virhetarkistukset Sample call: kutsuesimerkki Esimerkki /* Funktio: : maxi * Tehtävä: Etsii suuremman parametriarvon * Parametrit: kaksi kokonaislukua * Palauttaa: suuremman arvon * Muuttaa: ei mitää ään * Virhetark.: ei ole * Kutsuesimerkki: : i = maxi(k, 3) */ int maxi(int i, int j) return i > j? i : j; Esimerkki: sarjan summa /* Function: oneovernseries * Purpose: compute the sum of 1/N series * Inputs: n (parameter) * Returns: the sum of first n elements of * 1+ 1/2 + 1/3 + 1/n * Modifies: nothing * Error checking: returns 0 if n negative * Sample call: i = oneovernseries(100); */ double oneovernseries(int); 2

double oneovernseries(int n) double x; /* Check boundary conditions */ if(n <= 0) for(x = 0, i = 1; i < n; i++) x += 1/((double)i); return x; Moduulit ja modulaarinen ohjelmointi C C ei suoraan tue modulaarista (tai rakenteista) ohjelmointitapaa ei sisäkk kkäisiä funktioita globaalit muuttujat näkyvn kyvät t saman tiedoston niille funktioille, joiden esittely tai määrittely on vasta muuttujan määm äärittelyn jälkeen Moduulit C:ssä on ohjelmoijan kuitenkin mahdollista käsitellä tiedostoja siten, että yhteen kuuluvat elementit sijoitetaan samaan tiedostoon yhden abstraktin tietotyypin toteutus yhteen tiedostoon Moduulia voi jopa ajatella luokan kaltaisena elementtinä Esim. tiedostot lista.h ja lista.c lista.h tietorakenteiden kuvaus ja funktioiden esittelyt lista.c funktioiden toteutukset ja omien muuttujien esittelyt Moduulit: esittely vs. toteutus Moduulilla on aina esittely otsikkotiedostossa (*.h) ja toteutus erillisessä ohjelmatiedostossa (*.c) käyttäjä (eri moduuli, Joku funkt.) main.c esittely (otsikkotiedosto) list.h toteutus list.c Esimerkki: LINUXin prosessilista prev next prev next prev next Linuxissa on abstrakti tietorakenne kahteen suuntaan linkitetty lista (käytet ytetään n mm. prosessin kuvaajille) Listalle on määm ääritelty omat käsittelyrutiinit k (funktioita ja makroja) list_add(n,h), list_add_tail(n,h) list_del(p) list_empty(p), list_entry(p,t,f) list_for_each(p,h) Prosessilistan omat makrot: : SET_LINKS REMOVE_LINKS (kts( kts.. www-sivu http://lxr.linux.no lxr.linux.no/source/include/linux/list.hlist.h ) Muuttujien määm äärittelyt ja näkyvyyssäännöt #include <stdio.h> /* globaalit muuttujat tänne*/ double tulos; int main(int argc, char** argv) /* funktion muuttujat tänne */ for (i=0; i < 20; i++) /* Lohkon muuttujia */ int j; j = i*4; Lohkon sisäiset iset muuttujat Määritellään n lohkon alussa Eivät t näy n y lohkon ulkopuolelle Arvo ei säily s kutsukerrasta toiseen Lohko on mikä tahansa aaltosulkujen kattama ohjelman osa Funktiokin on lohko Globaalit muuttujat Määritellään n funktioiden ulkopuolella Näkyvät t myöhemmin määritellyille funktioille Arvo säilyy s suorituksen ajan 3

Poikkeamat näkyvyyssn kyvyyssäännöistä: Viittaaminen muualla määriteltyyn extern static int si; extern int ei; void f() int fi; static int sif; extern int eif; /* vältä tätä! */ void g(); static void h(); int eif; /* määritelty vasta täällä */ Määrittely on tällt llöin tyypillisesti jossain toisessa tiedostossa tai kirjastossa Samassa tiedostossa, mutta myöhemmin Vältä käyttöä!! Pyri määm äärittelemään muuttujat mahdollisimman paikallisesti, ja ainakin yhden tiedoston sisäll llä Poikkeamat säilyvyysss ilyvyyssäännöistä: Funktion sisäinen inen muuttuja pysyväksi ja piiloon - static static int si; extern int ei; void f() int fi; static int sif; extern int eif; /* vältä tätä! */ void g(); static void h(); int eif; /* määritelty vasta täällä */ static määreellä muuttujan arvo säilyy s suorituskerrasta toiseen. Toisaalta static määre rajoittaa kyseisen muuttujan tai funktion (eli tunnuksen) näkyvyyttä.. Sitä ei voi tämän n jälkeen j käyttk yttää muista tiedostoista käsin. k (Vrt. javan private ) Muuttujien sijoittelu muistiin Extern määre kertoo kääk ääntäjälle, että tässä kohtaa muuttujalle ei tarvitse varata tilaa, koska muuttuja on määritelty muualla Static määre kertoo kääk ääntäjälle, että paikalliselle muuttujalle on varattava tilaa pinon ulkopuolelta, koska arvon pitää säilyä Register määre kertoo kääk ääntäjälle, että muuttujaa käytetään n niin paljon, että sille kannattaisi varata oma rekisteri prosessorilta tässt ssä lohkossa Muuttujan esittely ilman määm äärettä,, ns. automaattinen tilanvaraus globaaleille muuttujille tilanvaraus kääk äännösaikana paikallisille muuttujille tilanvaraus pinosta suoritusaikana Rekursio Rekursiivinen funktio kirjoitetaan c:ssä ihan niin kuin muissakin kielissä Rekursion pysäytt yttämiseksi on rekursiivisen kutsun oltava jollakin tavalla ehdollinen. int digits ( int n) if (n <= 0) if (n/10 == 0) return 1 + digits (n/10); int digits ( int n) int count =0; if (n <= 0) do count ++; n /= 10; while ( n!=0); return count; Osoittimet (pointers( pointers) Osoitin on muuttuja, jonka arvona on toisen muuttujan osoite. Osoittimien käyttk yttö perustuu siihen, että useimpien tietokoneiden muisti voidaan kuvata ikää ään n kuin valtavana yksiulotteisena taulukkona. Kaikki ohjelmat ja data sijaitsevat tässt ssä taulukossa. Osoitin on oikeastaan indeksi johonkin kohtaan muistia. Osoitinmuuttuja on erää äänlainen viitta, jonka avulla päästään n käsiksi k toiseen muuttujaan. Vrt. Javan olioviite (joka oikeastaan on osoitin) TITO Xptr DC 0 X DC 12 LOAD R1, =X STORE R1, Xptr LOAD R2, X LOAD R3, @Xptr Tieto ja sen osoite (3) ; R1 230 muisti Xptr=225: 230 ; R2 12 12345 ; R3 12 12556 Muuttujan X osoite on 230 128765 Muuttujan X arvo on 12 12222 Osoitinmuuttujan (pointterin) Xptr osoite on 225 X=230: 12 Osoitinmuuttujan Xptr arvo on 230 12998 jonkun tiedon osoite (nyt X:n osoite) Osoitinmuuttujan Xptr osoittaman kokonaisluvun arvo on 12 C-kieli: Y = *ptrx /* ei prtx:n arvo, mutta ptrx:n osoittaman muuttujan arvo */ 4

int main(int argc, char** argv) int x=1, y=2, z[10]; int *ip; int *p, q; int *r, *s; ip=&x; y = *ip; /* y = x =1 */ *ip = 0; /* x = 0 */ ip = &z[0]; double atof(char *string); KOODI ESIMERKKI: HUOM: p on osoitin ja q tavallinen kokonaisluku x y z ip 1 2 0 1 Funktion parametrina osoitin on erittäin tavallinen Viiteparametrit Välitetään n muuttujan osoite (eli viite), jos funktion pitää muuttaa muuttujan arvoa Funktion esittelyssä muodollinen parametri on osoitin void lue_muuttujaan (int *luku); Funktion kutsussa todellisena parametrina välitetään n muuttujan osoite lue_muuttujaan (&x); (kun int x;) lue_muuttujaan (ptr); (kun int *ptr;) Viiteparametrin käyttö Esimerkkejä: funktion oma paluuarvo on tieto tehtävän onnistumisesta, tulos parametrissa /* Lue korkeintaan n merkkiä * Palauta merkin c esiintymisten lukumää äärä * Paluuarvona onnistumistieto! */ int readit(int n, char c, int *occurrences); /* Palauta lukujen n ja m summa ja tulo */ int compute(int n, int m, int *sum, int *product); Vaihteleva parametrimää äärä (vrt. printf, scanf) Yksi nimetty ja tyypitetty parametri, loput kolme pistettä Parametrien käsittely (va_list, va_start, va_arg) #include<stdarg.h stdarg.h> /* return a product of double arguments */ double product(int number,...) va_list list; double p; va_start(list (list,, number); for(i = 0, p = 1.0; i < number; i++) p *= va_arg(list (list,, double); va_end(list (list); return p; Parametrilistan tyyppi AINA: Haetaan nimetty Haetaan seuraava HUOM: Ei autom. tyyppimuunnoksia Vaihteleva parametrimää äärä OIKEIN: product(2, 2.0, 3.0) * product(1, 4.0, 5.0) VÄÄRIN: product(2, 3, 4) Vaihtelevassa parametrilistassa EI voi tehdä automaattista tyyppimuunnosta,, kun tyyppi ei ole tiedossa käännösaikana! Funktio-osoittimet osoittimet Vaikka funktiot eivät t ole muuttujia, niin niilläkin on osoite muistissa (kts( kts.. esim. TTK-91 konekieli). Tähän T n osoitteeseen voi viitata funktio-osoittimen osoittimen avulla. Osoitin määm ääritellään n seuraavasti paluutyyppi (*nimi) (parametrilista); int (*lfptr lfptr) ) (char( char[], int); lfptr = getline; /* kun int getline(char s[], int len); */ Funktio-osoittimen osoittimen avulla, voidaan eri kutsukerroilla kutsua eri funktiota. Näin N voidaan rakentaa yleisiä aliohjelmia. (Esim. stdlib.h:ssa on funktio qsort,, joka saa parametrina järjestysfunktion) 5

Funktio-osoittimien osoittimien käyttö parametrina Funktioita voi kutsua useita kertoja Funktion paluuarvo ja parametrilista on tyypitetty Geneerisissä tietorakenteissa ja niiden käsittelyrutiineissa käytetään usein tyypittömi miä parametreja ja paluuarvoja: Yleiskäytt yttöisempiä rutiineja, mutta paljon enemmän virhemahdollisuuksia Usein alkion sisält ltöä käsittelevä funktio välitetään funktioparametrina Esimerkki funktiotietue (kts. include/linux linux/quota.h) 231 /* Operations which must be implemented by each quota format */ 232 struct quota_format_ops 233 int (*check_quota_file)(struct super_block *sb, int type); /* Detect whether file is in our format */ 234 int (*read_file_info)(struct super_block *sb, int type); /* Read main info about file - called on quotaon() */ 235 int (*write_file_info)(struct super_block *sb, int type); /* Write main info about file */ 236 int (*free_file_info)(struct super_block *sb, int type); /* Called on quotaoff() */ 237 int (*read_dqblk)(struct dquot *dquot); /* Read structure for one user */ 238 int (*commit_dqblk)(struct dquot *dquot); /* Write structure for one user */ 239 int (*release_dqblk)(struct dquot *dquot); /* Called when last reference to dquot is being dropped */ 240 ; Ohjelmointiesimerkki Suunnittele ohjelma, joka lukee käyttk yttäjän syött ttämän n rivin. Rivin ensimmäinen inen merkki kertoo, mitä rivillä oleville kokonaisluvuille pitää tehdä. S - rivin luvuista pitää etsiä suurin P etsitää ään n pienin A - lasketaan lukujen summa K lasketaan keskiarvo Millainen rakenne Luonnollisesti Yhden luvun lukeminen omaan funktioon Pääohjelmaan tuo valinta Mutta entä yhden rivin käsittely? k Kullekin tyypille oma rivinkäsittelyfunktio ja valinta toimenpiteen mukaan? Geneerinen käsittelyfunktio, joka saa parametrina laskennan? Millainen rakenne? Verrataan! Kullekin oma: Geneerinen toisto: Tehdää ään n ensin: kullekin oma Funktiot: lue_luku,, main, pienin(), Lue merkki Switch merkki P :: pienin(); S: suurin(); A: summa(); K: keskiarvo(); Tulosta arvo; Lue merkki Switch merkki P : toistofunktio(pienin) S: toistofunktio(suurin) A: toistofunktio(summa) K : toistofunktio(keskiarvo) Tulosta arvo; /* tehdään erillinen lukufunktio, jota voidaan myöhemmin parantaa */ int lue_luku(int *luku) return scanf( %d, luku); int pienin() int luku, pieni = INT_MAX; while (lue_luku(&luku) == 1) if (luku < pieni ) pieni = luku; return pieni; int main () char merkki; int tulos; printf( Syötä tietoa yksi rivi\n ); while ( (merkki=getchar())!= EOF) switch (merkki) case P : tulos = pienin (); break; case S : tulos = suurin (); break; case A : tulos = summa(); break; case K : tulos = ka (); break; default : break; printf ( %c: %d\n, merkki, tulos); 6

Tehdää ään: geneerinen toisto Funktiot: lue_luku,, main, pienin(), int toisto(int (*fun) (int luku)) int luku, tulos; while (lue_luku(&luku) == 1) tulos = fun (luku); return tulos; int pienin( int luku) static int pieni = INT_MAX; if (luku < pieni) pieni = luku; return pienin; int main () char merkki; int tul; printf( Syötä tietoa yksi rivi\n ); while ( (merkki=getchar())!= EOF) switch (merkki) case P : tul = toisto(pienin); break; case S : tul = toisto(suurin); break; case A : tul = toisto(summa); break; case K : tul = toisto(ka); break; default : break; printf ( %c: %d\n, merkki, tul); Muutetaan tehtävää ää Entäpä jos pitäisikin isikin lukea useita rivejä? Kumpi ratkaisu soveltuisi tähän t n paremmin? Miksi? Geneerisen toiston saa soveltumaan, kun Static muuttujat pois funktioista (globaaleiksi) Kullekin muuttujalle alustusfunktio lisäksi Yleiskäytt yttöisyys lisää ääntyy, mutta Tämä on raskas ratkaisu näin n yksinkertaiseen ongelmaan Muutettu geneerinen toisto /* pieni.h */ void alustapieni (int a); int Pienin( int luku); static int pieni; void alustapieni (int a) pieni = a; return; int Pienin( int luku) if (luku < pieni) pieni = luku; return pienin; int main () char merkki; int tul; /* rivi toisto tähän ympärille*/ alustapieni(-1); alustasuurin(-int_max); printf( Syötä tietoa yksi rivi\n ); while ( (merkki=getchar())!= EOF) switch (merkki) case P : tul = toisto(pienin); break; case S : tul = toisto(suurin); break; case A : tul = toisto(summa); break; case K : tul = toisto(ka); break; default : break; printf ( %c: %d\n, merkki, tul); C assumes that programmer is intelligent enough to use all of its constructs wisely, and so few things are forbidden. C can be a very useful and elegant tool. People often dismiss C, claiming that it is responsible for a bad coding style. The bad coding style is not the fault of the language, but is controlled (and so caused) by the programmer. /* Search a block of double values */ int search( const double *block, size_t size, double value) double *p; if(block == NULL) for(p = block; p < block+size; p++) if(*p == value) Etsintä: funktio search Osoitin viiteparametrina Tietorakenteen läpikäynti Yleistetää ään tuo etsintäfunktio search C ei salli polymorfismia, mutta voimme simuloida sitä käyttämällä geneerisiä osoittimia (i.e. void*). Funktion esittelyssä voi määritellä että paluuarvo ja parametrit ovatkin määräämätöntä tyyppiä void Prototyyppi: alkioiden lkm vertailufunktio int searchgen(const void *block, size_t size, void *value); /* Määrittämätön tyyppi ei riitä*/ /* Vaan tarvitaan lisää parametreja */ int searchgen(const void *block, size_t size, void *value, size_t elsize int (*compare)(const void *, const void *)); tietorak. alkion koko 7

Funktion kutsujan toimenpiteet Funktion kutsujan täytyy määritellä vertailufunktio, jota voidaan kutsua etsintäfunktiosta (ns. Call back -rutiini) Tyyppien kanssa määrittely voisi olla: int comp(const double *x, const double *y) return *x == *y; Tyypittömien parametrien avulla funktio täytyykin määritellä seuraavasti int comp(const void *x, const void *y) return *(double*)x == *(double*)y; geneerisen etsinnän käyttö /* Application of a generic search */ #define SIZE 10 double *b; double v = 123.6; int main (void) if(malloc(b, double, SIZE)) exit(exit_failure); for(i = 0; i < SIZE; i++) /* initialize */ if(scanf("%lf", &b[i])!= 1) free(b); exit(exit_failure); printf("%f was %s one of the values\n", v, searchgen(b, SIZE, &v, sizeof(double), comp) == 1? "" : "not"); /* tai exit(exit_success); */ Geneerisen funktion search toteutus #define VOID(targ,size) ((void *)(((char *)(targ)) + \ (size))) int searchgen(const void *block, size_t size, void *value, size_t elsize, int (*compare)(const void *, const void *)) void *p; if(block == NULL) for(p = (void*)block; p< VOID(block,size*elsize); p = VOID(p,elsize)) if(compare(p, value)) HUOM: Osoittimen siirrossa on nyt otettava myös alkion koko huomioon! 8