Apua c-ohjelman kanssa!

aloitteleva_kdr

Teen viopessa c-ohjelmoinnin harjoituksia ja eräs tehtävä tuottaa hieman päänvaivaa. Ensimmäiset kaksi viopen testeistä menevät läpi mutta kolmas, joka on piilotettu feilaa.. Osaisiko joku jeesata ja kertoa mitä pitäisi muuttaa, että ohjelma toimisi. En siis tiedä mitä nimiä viope syöttää ohjelmalle, mutta lopputulos on seuraava:

Tiedoston luettelo.txt sisältö:

3
Keijo Kojootti 050-3500980
Brian Kottarainen 040-3980982
Brian Kottarainen 040-3980982

Kun tiedoston pitäisi olla seuraavanlainen:

3
Keijo Kojootti 050-3500980
Brian Kottarainen 040-3980982
Brita Kottarainen 05-4567393

Eli ongelma liittyy varmaankin tohon että molemmat Brian ja Brita alkaa samoilla kirjaimilla ja sukunimi on sama. En vaan keksi mitä pitäisi muuttaa. Pistän alle tehtävänannon ja lopuksi vielä koodin.

Tehtävänanto:
Puhelinluettelo - henkilön tietojen poistaminen ohjelmointitehtävä 3/3
Tee ohjelma, jolla voidaan poistaa luettelosta tietyn henkilön tiedot. Ohjelma kysyy ensin etunimen ja tämän jälkeen sukunimen. Ohjelma tarkistaa, löytyykö henkilön tiedot. Mikäli henkilö löytyy luettelosta, tiedot poistetaan. Muuten tulostetaan virheilmoitus "Etsimääsi henkilöä ei löytynyt luettelosta.".

Vihje:
Ennen kuin ryhdyt laatimaan varsinaista koodia, sinun kannattaa suunnitella ohjelma hyvin. Rakenteiden käyttäminen voi helpottaa tehtävää. Päivitetyn listan tallentaminen mitä ilmeisimmin tuhoaa aikaisemmat tiedot tiedostosta luettelo.txt.

Nimitietojen vertailu onnistuu c-kielen funktiolla strcmp(), joka löytyy kirjastotiedostosta string.h. Voit tutustua funktion käyttöön c-manuaalissa.
Esimerkkitulostus


Anna etunimi:Timo
Anna sukunimi:Timpuri
Tiedot poistettu luettelosta.

10

2236

    Vastaukset

    Anonyymi (Kirjaudu / Rekisteröidy)
    5000
    • aloitteleva_kdr

      KOODI:

      #include
      #include
      #include

      struct henkilo {
      char enimi[21];
      char snimi[21];
      char puh[21];
      };

      const char rekisteri[] = "luettelo.txt";

      char* TiedonKopiointi(char*out, char*in, unsigned koko)
      {
      char* pc, *pc2;
      unsigned j;
      memset(out, 0, koko);
      pc = in;
      pc2 = out;
      for(j = koko; j && *pc != ' ' && *pc != '\n'; --j){
      *pc2 = *pc;
      pc2 ;
      pc ;
      }
      return pc;
      }

      struct henkilo* TaytaTietue(char*buffer, struct henkilo*phenk)
      {
      char* pc;
      pc = buffer;
      pc = TiedonKopiointi(phenk->enimi, pc, 21);
      pc = TiedonKopiointi(phenk->snimi, pc, 21);
      pc = TiedonKopiointi(phenk->puh, pc, 21);
      return phenk;
      }

      unsigned LueTiedostoMuistiin(struct henkilo*henkilo_lista)
      {
      char buffer[sizeof(struct henkilo)];
      FILE *tiedosto;
      int hlomaara = 0;
      struct henkilo* phenk;
      long i;
      if ((tiedosto = fopen(rekisteri, "r")) != NULL) {
      fgets(buffer, 4, tiedosto);
      hlomaara = atoi(buffer);
      if (hlomaara >= 50) {
      printf("Korkeintaan %d tietuetta", 50);
      fclose(tiedosto);
      return -1;
      }
      for(i = 1, phenk = henkilo_lista; i i)
      return phenk;
      else if(poistettava != i){ /*Viimeistä poistetaan*/
      phenk1 = henkilo_lista;
      phenk1 = (poistettava - 1); /*Poistettavan kohdalle*/
      phenk2 = phenk1;
      phenk2 ; /*Ensimmäisen siirrettävän kohdalle*/
      /*Siirretään loppuja pykälällä*/
      pc = (char*)(void*)phenk2;
      pc2 = (char*)(void*)phenk;
      memcpy(phenk1, phenk2, pc - pc2);
      }
      *hlomaara = i - 1;
      return --phenk;
      }

      unsigned TallennaTiedostoon(struct henkilo* phenk, unsigned maara)
      {
      FILE *tiedosto;
      unsigned i;
      if ((tiedosto = fopen(rekisteri, "w")) == NULL) { /*Uuden tiedoston luonti*/
      printf("Virhe tiedoston avaamisessa.");
      return 1;
      }
      fprintf(tiedosto, "%d\n", maara); /*Kirjoitettavien määrä tiedoston alkuun*/
      for(i = maara; i; --i){
      fprintf(tiedosto,"%s %s %s\n", phenk->enimi, phenk->snimi, phenk->puh);
      phenk ;
      }
      fclose(tiedosto);
      return 0;
      }

      int main(void)
      {
      struct henkilo henkilo_lista[50];
      unsigned hlomaara, poistettava;
      struct henkilo* phenk;
      unsigned i, j;
      struct henkilo etsittavahlo;

      hlomaara = LueTiedostoMuistiin(henkilo_lista);
      if(hlomaara > (unsigned)-1)
      return 1;

      printf("Anna etunimi:");
      scanf("%s", etsittavahlo.enimi);
      printf("Anna sukunimi:");
      scanf("%s", etsittavahlo.snimi);

      for(i = 0, j = 0, phenk = henkilo_lista; i < hlomaara ; phenk , i){
      if(strcmp(phenk->enimi, etsittavahlo.enimi) == 0) {
      if(strcmp(phenk->snimi, etsittavahlo.snimi) == 0) {
      PoistaHenkilo(henkilo_lista, phenk, &hlomaara, i 1);
      printf("Tiedot poistettu luettelosta.");
      j = 1;
      break;
      }
      }
      }
      if(!j)
      printf("Etsimääsi henkilöä ei löytynyt luettelosta.");

      /*Tallennetaan lista tiedostoon*/
      TallennaTiedostoon(henkilo_lista, hlomaara);

      return 0;
      }

    • johanonnistuu.

      tuttujuttu, et käytä IBM ASCII-järjestelmää, jossa on kaikki ääkköset ja öökköset, vaan uutta "muka uutta (25v vanhaa)", tai toinen vaihtoehto on nämä "whitespacet", eli.. joka tapauksessa kirjaimet toimivat, mutta huono uusi näppäinjärjestelmä.

      Tuon voi kyllä kiertää jos osaa, laitat vaan käyttöjärjestelmään että IBM 256-letter ASCII.

      • jf394yfgefok4

        Mutta voisin mainita tuosta sinun c-koodista... näyttääpä sotkuiselta, koneella tehtyä jotain :) tuonkin voisit korjata pääfunktiossa: "void main(void)", ja palautus "return ;"


      • sdfsfsdfs54353
        jf394yfgefok4 kirjoitti:

        Mutta voisin mainita tuosta sinun c-koodista... näyttääpä sotkuiselta, koneella tehtyä jotain :) tuonkin voisit korjata pääfunktiossa: "void main(void)", ja palautus "return ;"

        Kai tuo oli vitsi.

        void main(void); on totaalisen väärin.

        int main(void); on oikea standardin mukainen menettely.


    • idomaattinen9

      Ei ole vaan täysin oikein :)
      ei oteta vastaan mitään, eikä palauteta mitään koko ohjelmasta :)

      Muuten näyttää tuo hyvälle ihan kun tarkemmin katoin, mutta miksi palautat koko ohjelmasta mitään, kun kerran jo tallennat tulokset tiedostoon???

      siitä tämä void-juttu.

      Tuo puuttuu sulla tiedostoon tulostamisesta: "\r \n", kumpikin pitää olla peräkkäin..

    • tyhmättyhjätincludet

      Myöskin jos tuo on C/C kääntäjä mitä käytät, niin lisätä noiden tyhjien include-tiedostojen tilalle: "using namespace std;"

    • noohhhhhflds43lf43

      void main(void), sitten return (void); täysin oikein.

      • sdfsdfsdfsd

        Tuo on totaalisen väärin.

        Ensinnäkin standardin mukaan mainin täytyy olla aina INT tyyppinen ja mainin täytyy aina palauttaa INT arvo.

        Tämä siksi että käyttiksen puolella voidaan havaita mitä ohjelma palauttaa ja sen perusteella päätellä onnistuiko ohjelman suoritus ja mitä virheitä ohjelmassa tapahtui.

        Tuo void main(void); käytäntö on jostian kivikaudelta kun C:n standardi ei ollut vielä kehittynyt.

        Muistakaa AINA: "int main(void)" tai "int main(int argc, char *argv[])"
        Nämä ovat ainoat standardinmukaiset vaihtoehdot puhtaassa C:ssä.

        Toki myös on winmaineja muita entrypointteja mutta ne toteuttavat saman kaavan.


    • Eiköhän opiskelun tarkoitus ole itse oppia näin yksinkertainen ohjelma, eikä yrittää saada joku netistä korjaamaan ohjelmaasi. Vilkaisepa sitä toisen kerran paremmin. Kyllä se syy selkeästi on koodissasi.

      Se mitä standardia main function kanssa käytetään on täysin jonni joutavaa kiistelyä, joka viittaa siihen, että kiistelijä on joko kaikkea vihaava ideologi tai melko kokematon ohjelmoija. Ei ole merkitystä, millä standardilla se on kirjoitettu, kunhan se toimii oikein.

      • tgdgdfgdfg

        Ei ole jonninjoutavaa.

        Ensinnäkin miksi opetalla huonoja ja VÄÄRIÄ ohjelmointitapoja kun voi samalla vaivalla opetella asian oikein.

        Lisäksi nimenomaan "void main()" version kanssa tukee oikeasti ongelmiakin käyttisalustoilla. Sulautetussa void main(); voi olla ok jos alusta ei vaadi / tarkista mitä main palauttaa.


        Tässä syitä käyttää standardinmukaista oikeaoppista int mainia.

        1. Koska se on standardi

        2. Koska ohjelman aloitus rutiinit jotka kutsuvat mainia saattavat olettaa tai olettavat että mainin palautusarvo asetetaan pinoon. Jos main funktio ei tee tätä, saattaa ohjelman lopetus rutiinin pino vikaantua / korruptoitua ja aiheuttaa määrittelemätöntä toimintaa tai kaatumisia.

        3. Koska standardista poikkeava maini saattaa palauttaa aivan sattumanvaraisen arvon kutsuvalle käyttikselle. Tämän takia ei pystytä tarkistamaan esimerkiki ohjelman vikakoodia. Lisäksi makefilet ja vastaavat luottavat ohjelmien palautusarvoihin toiminnoissaan.

        4. Koska mainin palauttama kokonaislukua käytetään ohjelman tilan tarkastuun (onnistunut suoritus, virhetilanne, virhekoodi).


    Ketjusta on poistettu 0 sääntöjenvastaista viestiä.

    Luetuimmat keskustelut

    1. Aivosyöpää sairastava Olga Temonen TV:ssä - Viimeinen Perjantai-keskusteluohjelma ulos

      Näyttelijä-yrittäjä Olga Temonen sairastaa neljännen asteen glioomaa eli aivosyöpää, jota ei ole mahdollista leikata. Hä
      Maailman menoa
      91
      2910
    2. Pelotelkaa niin paljon kuin sielu sietää.

      Mutta ei mene perille asti. Miksi Venäjä hyökkäisi Suomeen? No, tottahan se tietenkin on jos Suomi joka ei ole edes soda
      Maailman menoa
      299
      1686
    3. Mikä saa ihmisen tekemään tällaista?

      Onko se huomatuksi tulemisen tarve tosiaan niin iso tarve, että nuoruuttaan ja tietämättömyyttään pilataan loppuelämä?
      Sinkut
      246
      1557
    4. Minkä merkkisellä

      Autolla kaivattusi ajaa? Mies jota kaipaan ajaa Mersulla.
      Ikävä
      87
      1391
    5. IL - VARUSMIEHIÄ lähetetään jatkossa NATO-tehtäviin ulkomaille!

      Suomen puolustuksen uudet linjaukset: Varusmiehiä suunnitellaan Nato-tehtäviin Puolustusministeri Antti Häkkänen esittel
      Maailman menoa
      402
      1384
    6. Nyt kun Pride on ohi 3.0

      Edelliset kaksi ketjua tuli täyteen. Pidetään siis edelleen tämä asia esillä. Raamattu opettaa johdonmukaisesti, että
      Luterilaisuus
      400
      1307
    7. Kiitos nainen

      Kuitenkin. Olet sitten ajanmerkkinä. Tuskin enää sinua näen ja huomasitko, että olit siinä viimeisen kerran samassa paik
      Tunteet
      2
      1089
    8. Esko Eerikäinen tatuoi kasvoihinsa rakkaan nimen - Kärkäs kommentti "Ritvasta" lävähti somessa

      Ohhoh! Esko Eerikäinen on ottanut uuden tatuoinnin. Kyseessä ei ole mikä tahansa kuva minne tahansa, vaan Eerikäisen tat
      Suomalaiset julkkikset
      38
      1037
    9. Hyväksytkö sinä sen että päättäjämme ei rakenna rauhaa Venäjän kanssa?

      Vielä kun sota ehkäpä voitaisiin välttää rauhanponnisteluilla niin millä verukkeella voidaan sanoa että on hyvä asia kun
      Maailman menoa
      329
      874
    10. Miksi Purra-graffiti ei nyt olekkaan naisvihaa?

      "Pohtikaapa reaktiota, jos vastaava graffiti olisi tehty Sanna Marinista", kysyy Tere Sammallahti. Helsingin Suvilahden
      Maailman menoa
      257
      855
    Aihe