Frage zu getline



  • Die gesamte Funktion sieht so aus.

    int get_Mat()
    {
    
    	int max_columns=0;
    	int row_counter = 0;
    	int column_counter = 0;
    	vector<double> value;
    
    	infile.open("D:\\petersp\\Master-Arbeit DLD\\PF_2D\\properties_C35.txt", ios::in);
    
    	if (infile.is_open())
    	{
    		while(!infile.eof())
    		{
    			getline(infile, s);
    
    			cout << "Aktuelle Position: " << infile.tellg() << "\n";
    
    			row_counter++;
    			size_t pos = s.find("\t");
    			column_counter = 0;
    
    			while(pos != s.npos)
    			{
    				string strpart = s.substr(0, pos);
    
    				cout << "Länge von s: " << s.length() << "\n";
    
    				s = s.substr(pos+1,s.length());
    				value.push_back(strtod(strpart.c_str(), NULL));
    				column_counter++;
    				pos = s.find("\t");
    			}
    			column_counter++;
    			value.push_back(strtod(s.c_str(), NULL));
    			if (column_counter > max_columns)
    			{
    				max_columns = column_counter;
    			}
    
    			cout << value[1] << "   " << value[2] << "\n";
    		}
    
    		Mat.resize(row_counter);
    		for (unsigned i = 0; i < Mat.size(); ++i)
    			Mat[i].resize(max_columns);
    
    		infile.seekg(0);
    		infile.clear();
    
    		while(!infile.eof())
    		{
    
    			getline(infile, s);
    			size_t length = s.length();
    			int i=0;
    			size_t pos = s.find("\t");
    
    			while(pos != s.npos)
    			{
    				string strpart = s.substr(1, pos-1);
    				s = s.substr(pos+1,length);
    				value.push_back(strtod(strpart.c_str(), NULL));
    				length = s.length();
    				for(int j=0; j<max_columns;j++)
    				{
    					//cout << "Fülle Matrix\n";
    					Mat[i][j]=value[j];
    					cout << "Mat(" << i << "," << j << ") = " << Mat[i][j] << "\n";
    				}
    				i++;
    			}
    
    		}
    
    		infile.close();
                    return 1;
    	}
    	else
    	{
    		return 0;
    	}
    
    }
    

    Bei Mat handelt es sich um eine globale Variable. Wie gesagt weiss ich, dass es nicht gerade elegant programmiert ist. Das ist jedoch auch keine Anforderung.

    Ich verstehe jedoch nicht warum es nicht funktioniert.

    Aus der cplusplus.com Reference:

    If the delimiter is found, it is extracted and discarded, i.e. it is not stored and the next input operation will begin after it.

    Das bedeutet doch, es müsste bei einem zweiten Aufruf von getline die nächste Zeile eingelesen werden.

    Es handelt sich auch nicht um eine Hausaufgabe. Ich versuche im Rahmen miner Master-Arbeit ein an unserem Institut entwickeltes FE-Tool (Finite Elemente) zu erweitern, um damit einen neuartigen Umformprozess simulieren zu können. Dabei stolpere ich jedoch immer wieder über C++-Probleme, da ich mich damit leider nicht wirklich gut auskenne. (Bin dabei das zu ändern 😉 ) Es fällt mir dabei nicht immer leicht, abzuschätzen ob meine Frage jetzt "doof" ist, bzw. den Anforderungen dieses Forums genügt. Falls sich jemand über meine Fragen ärgert, bitte ich denjenigen, mir das mitzuteilen. Ich will niemandem auf die Nerven gehen.



  • Habs mittlerweile selber rausgefunden. Liegt am pushback, ich schreibe immer die ersten werte von values in das array. Das ist dann wohl ein Bsp. für eine doof Frage 🙄



  • pitaz schrieb:

    Habs mittlerweile selber rausgefunden. Liegt am pushback, ich schreibe immer die ersten werte von values in das array. Das ist dann wohl ein Bsp. für eine doof Frage 🙄

    Ich bezweifel das was auch immer am push_back liegt. Eine Bessere Fehlerbeschreibung wäre auch hilfreicher ("Ich verstehe jedoch nicht warum es nicht funktioniert." Ist KEINE sinnvolle Fehlerbeschreibung).

    Und wenn du wie du sagst "im Rahmen meiner Master-Arbeit" das ganze schreibst, sollte ich dir zutrauen können, auch sinnvolle und vor allem ausreichend formulierte Fehlermeldung (wenn möglich mit angabe der ersten Compilerfehler, falls es welche gibt...) zu verfassen.

    Mich wundert es auch nicht das du mit C++ Probleme hast. Du programmierst scheinbar in C++ auf eine C-"Art und Weise", die Konzepte unterscheiden sich aber wesentlich.

    cu André
    P.S: Und das dein Code nicht "gerade elegant programmiert ist" brauchst du nicht mal zu schreiben 😉



  • pitaz schrieb:

    Habe folgenden Code, der zugegebenermasse etwas umständlich die Anzahl Zeilen und Anzahl Spalten der Tabelle im txt-File ermittelt. Das funktioniert soweit auch. Jedoch liest getline bei jedem Durchlauf der while -Schleife die erste Zeile ein, obwohl die Position des get-Pointers jedes mal zur nächsten Zeile geht.

    das getline arbeitet korrekt, aber der vector value wird immer weiter geschrieben, während Du immer wieder das 2. und 3. Element ausgibst (warum nicht das erste?), welche nie überschrieben werden.

    Verschiebe die Zeile 7 vor die Zeile 15, und kopiere die gleiche Zeile nochmal vor die Zeile 57.
    Füge dann vor Zeile 43 noch ein 'if( value.size() > 2 )' ein.
    Tausche die Zeilen 50 und 51 aus. Man kann den seekg nur dann erfolgreich aufrufen, wenn der Stream vorher nicht im Fehlerzustand war.
    Den Code ab Zeile 62 solltest Du nochmal überarbeiten, da sind die Schleifen durcheinander gekommen.
    ggf. die Zeile 68 noch gegen 'for(int j=0; j<value.size(); ++j)' austauschen, dann stürzt es wenigstens nicht ab.

    :xmas2: Werner



  • Hallo nochmal

    Es hat nur daran gelegen dass ich den vector value nicht vor jeder schleife wieder geleert hab. Jetzt läuft es ganu so wie ich mir das vorstelle.

    asc schrieb:

    Mich wundert es auch nicht das du mit C++ Probleme hast. Du programmierst scheinbar in C++ auf eine C-"Art und Weise", die Konzepte unterscheiden sich aber wesentlich.

    Leider weiss ich nicht was C++ auf eine C-Art ist. Bin halt kein Programmierer. Hatte vor Jahren mal nen c++-kurs an der Uni und versuche nun, dieses (sehr beschränkte) Wissen wieder hervorzuholen.



  • pitaz schrieb:

    Leider weiss ich nicht was C++ auf eine C-Art ist. Bin halt kein Programmierer. Hatte vor Jahren mal nen c++-kurs an der Uni und versuche nun, dieses (sehr beschränkte) Wissen wieder hervorzuholen.

    Naja, C++ basiert ja auf C, trotzdem sind mit der Objektorientierung und generischen Programmierung völlig neue Paradigmen ins Spiel gekommen.

    Dein Code ist aber nicht typisch für ein "C in C++ programmieren" - das einzige, was ich beim Überfliegen gesehen habe, war strtod() , was du über std::stringstream lösen könntest. Und rohe Arrays sind halt nicht so schön.

    Für Tutorials und gute Bücher zu C++ schaust du am besten in den FAQ nach.



  • Sorry, aber was genau ist ein roher array?

    Du mienst warscheinlich die Mat Variable.

    Hab sie so initialisiert: vector<vector<double>> Mat;

    Ist das nicht ok?

    Wenn nicht, warum nicht?



  • ein rohes array (auch oft als C-Array bezeichnet) ist so was:

    statisch:
    Type name[anz];

    dynamisch:
    Type *name = new Type[anz];
    (nicht zu vergessen delete []name; - wobei wir auch schon beim nachteil angekommen wären ^^)

    std::vector ist ein wrapper für solch rohe arrays (stellt zuweisungs/vergleichs/... -operatoren bereit und kümmert sich um (de)allozierung(en)) und ist - so lange nichts dagegen spricht - den rohen Arrays vorzuziehen.

    btw:
    in deinem code wäre ein reserve und push_back besser als ein resize und [] so fern ich jz nichts übersehen habe...

    bb



  • pitaz schrieb:

    Ist das nicht ok?

    Doch, sorry, ich hatte falsch nachgeschaut. Tut mir leid.

    unskilled schrieb:

    std::vector ist ein wrapper für solch rohe arrays [...] und ist - so lange nichts dagegen spricht - den rohen Arrays vorzuziehen.

    Nicht ganz. Um genau zu sein, ist std::vector nur ein Wrapper für dynamische Arrays. 😉

    Für statische Arrays ist std::tr1::array (oder boost::array ) besser geeignet.



  • Nexus schrieb:

    unskilled schrieb:

    std::vector ist ein wrapper für solch rohe arrays [...] und ist - so lange nichts dagegen spricht - den rohen Arrays vorzuziehen.

    Nicht ganz. Um genau zu sein, ist std::vector nur ein Wrapper für dynamische Arrays. 😉

    Für statische Arrays ist std::tr1::array (oder boost::array ) besser geeignet.

    sry, dass ich das net mit erwähnt hatte - lag daran, dass ich das statische array erst ergänzt hab, als ich im prinzip schon fertig mit schreiben war ^^

    bb



  • Noch was

    Nexus schrieb:

    Dein Code ist aber nicht typisch für ein "C in C++ programmieren" - das einzige, was ich beim Überfliegen gesehen habe, war strtod() , was du über std::stringstream lösen könntest. Und rohe Arrays sind halt nicht so schön.

    Ich weiss, dass die strtod-Funktion nicht C++ ist. Hab aberin der string-Klasse keine Funktion gefunden. Wie genau konvertiert man denn einen string zu einem double in C++?



  • Eben mit std::stringstream .

    std::string StringVar;
    double DoubleVar;
    
    std::stringstream Stream;
    Stream << StringVar;
    Stream >> DoubleVar;
    

    Wenn du Boost hast, kannst du auch boost::lexical_cast benutzen.


Anmelden zum Antworten