簡體   English   中英

我沒有得到C ++中的分段錯誤

[英]segmentation fault in C++ I don't get

以下程序在運行時導致分段錯誤:

#include <iostream>
#include <fstream>
#include <cstring>
#include <iomanip>

using namespace std;

struct schokolade
{
    char name[20];
    int gewicht;
    int zutat_id[5];
    int menge_in_prozent[5];
};

struct zutat
{
    char name[20];
    float preis_pro_100gramm;
};


int main()
{
    char data[20];
    schokolade schokosorten[3];
    zutat zutaten[7];

    ifstream fin;
    fin.open("schoki.txt");
    //Check for error:
    if(fin.fail()){
        cout << "Datei schoki.txt konnte nicht geöffnet werden." << endl;
        return 0;
    }


    int anzahl;
    char name_zutat[20];
    int anteil;


    while(fin.getline(data,20)){

        if(strcmp(data, "vollmilch_mandel")==0){
            strcpy(schokosorten[0].name, data);
            fin >> schokosorten[0].gewicht;
            fin >> anzahl;

            for(int i=0;i<anzahl;i++){
                fin >> name_zutat >> anteil;
                if(strcmp(name_zutat, "mandeln")==0){
                    schokosorten[0].zutat_id[i]=0;
                    schokosorten[0].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "marzipan")==0){
                    schokosorten[0].zutat_id[i]=1;
                    schokosorten[0].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "milchpulver")==0){
                    schokosorten[0].zutat_id[i]=2;
                    schokosorten[0].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "kakao")==0){
                    schokosorten[0].zutat_id[i]=3;
                    schokosorten[0].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "kakaobutter")==0){
                    schokosorten[0].zutat_id[i]=4;
                    schokosorten[0].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "keks")==0){
                    schokosorten[0].zutat_id[i]=5;
                    schokosorten[0].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "zucker")==0){
                    schokosorten[0].zutat_id[i]=6;
                    schokosorten[0].menge_in_prozent[i]= anteil;
                    continue;
                }

            }

        }

        if(strcmp(data, "marzipan")==0){
            strcpy(schokosorten[1].name, data);
            fin >> schokosorten[1].gewicht;
            fin >> anzahl;

            for(int i=0;i<anzahl;i++){
                fin >> name_zutat >> anteil;
                if(strcmp(name_zutat, "mandeln")==0){
                    schokosorten[1].zutat_id[i]=0;
                    schokosorten[1].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "marzipan")==0){
                    schokosorten[1].zutat_id[i]=1;
                    schokosorten[1].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "milchpulver")==0){
                    schokosorten[1].zutat_id[i]=2;
                    schokosorten[1].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "kakao")==0){
                    schokosorten[1].zutat_id[i]=3;
                    schokosorten[1].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "kakaobutter")==0){
                    schokosorten[1].zutat_id[i]=4;
                    schokosorten[1].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "keks")==0){
                    schokosorten[1].zutat_id[i]=5;
                    schokosorten[1].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "zucker")==0){
                    schokosorten[1].zutat_id[i]=6;
                    schokosorten[1].menge_in_prozent[i]= anteil;
                    continue;
                }

            }
            schokosorten[1].zutat_id[anzahl]=-1;
        }

        if(strcmp(data, "keksschoki")==0){
            strcpy(schokosorten[2].name, data);
            fin >> schokosorten[2].gewicht;
            fin >> anzahl;

            for(int i=0;i<anzahl;i++){
                fin >> name_zutat >> anteil;
                if(strcmp(name_zutat, "mandeln")==0){
                    schokosorten[2].zutat_id[i]=0;
                    schokosorten[2].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "marzipan")==0){
                    schokosorten[2].zutat_id[i]=1;
                    schokosorten[2].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "milchpulver")==0){
                    schokosorten[2].zutat_id[i]=2;
                    schokosorten[2].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "kakao")==0){
                    schokosorten[2].zutat_id[i]=3;
                    schokosorten[2].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "kakaobutter")==0){
                    schokosorten[2].zutat_id[i]=4;
                    schokosorten[2].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "keks")==0){
                    schokosorten[2].zutat_id[i]=5;
                    schokosorten[2].menge_in_prozent[i]= anteil;
                    continue;
                }
                if(strcmp(name_zutat, "zucker")==0){
                    schokosorten[2].zutat_id[i]=6;
                    schokosorten[2].menge_in_prozent[i]= anteil;
                    continue;
                }

            }

        }

    }

    fin.close();


    fin.open("zutaten.txt");
    //Check for error:
    if(fin.fail()){
        cout << "Datei zutaten.txt konnte nicht geöffnet werden." << endl;
        return 0;
    }

    for(int i=0;i<7;i++){
        fin >> zutaten[i].name;
        fin >> zutaten[i].preis_pro_100gramm;
    }

    fin.close();

//*******************Problem area starts here **************************
    float R_Kosten_0 = 0.0;

    for(int i=0; i<5 && schokosorten[0].zutat_id[i]!=-1; i++){
        R_Kosten_0 += zutaten[schokosorten[0].zutat_id[i]].preis_pro_100gramm * schokosorten[0].menge_in_prozent[i]/100.0;
    }

    float R_Kosten_1 = 0.0;

    for(int i=0; i<5 && schokosorten[1].zutat_id[i]!=-1; i++){
        R_Kosten_1 += zutaten[schokosorten[1].zutat_id[i]].preis_pro_100gramm * schokosorten[1].menge_in_prozent[i]/100.0;
    }

    float R_Kosten_2 = 0.0;

    for(int i=0; i<5 && schokosorten[2].zutat_id[i]!=-1; i++){
        R_Kosten_2 += zutaten[schokosorten[2].zutat_id[i]].preis_pro_100gramm * schokosorten[2].menge_in_prozent[i]/100.0;
    }
 //*******************Problem area ends here **************************

    return 0;
}

我首先嘗試在其上使用valgrind,這給了我以下提示:

Conditional jump or move depends on uninitialised value(s)
==21457==    at 0x401B4E: main (in /root/Desktop/Beleg2)
==21457==  Uninitialised value was created by a stack allocation
==21457==    at 0x401108: main (in /root/Desktop/Beleg2)
==21457== 
==21457== Use of uninitialised value of size 8
==21457==    at 0x401AF7: main (in /root/Desktop/Beleg2)
==21457==  Uninitialised value was created by a stack allocation
==21457==    at 0x401108: main (in /root/Desktop/Beleg2)
==21457== 
==21457== Invalid read of size 4
==21457==    at 0x401AF7: main (in /root/Desktop/Beleg2)
==21457==  Address 0x105f15c5a4 is not stack'd, malloc'd or (recently) free'd

我不理解這些提示,因此我先后注釋掉了代碼的某些部分。 通過這種方法,我得出的結論是,分段錯誤必須在我標記過的代碼的特定部分中引起(請參見上文)。

但是,我只是不明白代碼的這一部分出了什么問題。 對我來說,看起來很好。

你們中有人知道這里發生了什么嗎?


這是文件schoki.txt的內容:

vollmilch_mandel
100
5
kakao 34
kakaobutter 10
milchpulver 20
zucker 28
mandeln 8

marzipan
100
3
kakao 50
zucker 25
marzipan 25

keksschoki
100
5
kakao 40
mandeln 7
kakaobutter 8
zucker 30
keks 15

這是文件zutaten.txt的內容:

mandeln 1.10
marzipan 1.22
milchpulver 0.6
kakao 0.82
kakaobutter 2.2
keks 0.64
zucker 0.14

如果在使用結構之前確保將其初始化為0,則效果會更好。

schokolade schokosorten[3];
zutat zutaten[7];

最簡單的方法是在結構中添加一個默認構造函數,以初始化所有成員

例如

struct zutat
{
    char name[20];
    float preis_pro_100gramm;
    zutat() : name(""),preis_pro_100gramm(0.0f) 
    {}
};

您正在加載文件,並假定內容將與您的結構匹配,但是如果其中之一丟失,則它將保持未初始化狀態。

您還應該進行一些邊界檢查,例如確保anzahl小於數組。

另一件事:將您的函數拆分為多個函數是一件好事,它使閱讀更容易且不易出錯。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM