Funktion mit 2 Rückgabewerten
-
Hallo,
Ich habe folgende Funktion:
(Ich muss das ganze mal auf GetLogicalDrives() umbauen, das ist bestimmt weniger Code, ändert aber an dem Problem nichts.)int find_used_dl(){ int dl_cnt = 0; // Zählt die Anzahl der gefunden Laufwerke CString dl[26]; // Array für A-Z CString result[26][2]; // Ergebnissarray int result_cnt = 0; // Arrayzähler int drivetype; // Laufwerkstyp int cnt = 65; // Schleifenzähler for(cnt; cnt<(65+26); cnt++ ) { // Wiederhole für jeden Laufwerksbuchstaben dl[cnt-65] = cnt; drivetype = GetDriveType( dl[cnt-65] + ":" ); if (drivetype != 1){ result[result_cnt][0] = dl[cnt-65]; result[result_cnt][1] = drivetype; result_cnt++; dl_cnt++; cout << "find_used_dl(): Habe Laufwerk " << dl[cnt-65] << ": gefunden! Typ: " << drivetype << "\n"; } } return dl_cnt; }Die Funktion gibt per return die Anzahl der verwendeten Laufwerke zurück.
z.B. Festplatte C: und
mit CD-ROM X: gibt 3.Nun will ich aber auch result[][] haben.
Dazu einige Verständnissfragen:
1. Was habe ich da mit [][] erzeugt?
In der Perl-Welt würde ich sagen Hash...?
Die Idee sieht so aus:
result[0][0] = "C";
result[0][1] = 3;
result[1][0] = "D";
result[1][1] = 3
result[2][0] = "X";
result[2][1] = 5;
Welchen Datentyp sollte result[][] haben? Ist CString ok?2. Die Funktion gibt im Moment die Anzahl der gefunden Laufwerke aus.
Allergings muss result[][] ja auch ausgegeben werden.
Dass man mit return nur 1 Wert zurückliefern kann, ist mir klar.
Folglich muss ich einen Pointer als Parameter übergeben (richtig?)
Hab das mal mit 3 int-Werten getestet, das hat funktioniert, und die int Werte wurden auch in der main-Funktion geändert.
Allerdings habe ich mit einem Pointer auf ein Array nur Fehlermeldungen geerntet.
Warscheinlich weil ich keine Ahnung von den ganzen Datentypen in C++ habe... Bin eben Perl-Umsteiger und tue mich mit den ganzen Datentypen wahnsinnig schwer...mfg
trequ
-
Zwei Rückgabewerte gehen nicht.
Du kannst aber mit Referenzen und Zeigern arbeiten, die du als Parameter übergibst.Als Datentyp für das Array würde ich BYTE vorschlagen. Da drin kann man Zahlen und einzelne Zeichen speichern.
Den Schleifenzähler kannst du gleich im Kopf eintragen.
dl_cnt ist das selbe wie result_cnt - kann man also weglassen.Belege vor dem Aufrufen den Speicherplatz für result!!!!!!!!! Das sieht genauso aus, wie du schon hattest. Der Funktion musst du nur den Namen des Array übergeben.
int find_used_dl(BYTE* result) { int result_cnt = 0; // Arrayzähler for(int cnt = 0; cnt < 26; cnt++ ) { // Wiederhole für jeden Laufwerksbuchstaben CHAR dl = cnt + 65; // Damit kriegt man den Buchstaben int drivetype = GetDriveType( dl + ":" ); if (drivetype != 1) { result[result_cnt][0] = dl; result[result_cnt][1] = drivetype; result_cnt++; cout << "find_used_dl(): Habe Laufwerk " << dl << ": gefunden! Typ: " << drivetype << "\n"; } } return result_cnt; }So, ich habe das jetzt weder getestet noch kompiliert - also sag, wenn es nicht geht.

Oder guck selbst auf dein Original, was ich da anders gemacht habe.Nachtrag:
Da ich mich mit Zeigern auch nicht gerade gut auskenne, hier noch ein Vorschlag:
Nimm ein CArray. Ja, das ist Speicherplatzverschwendung, aber es ist einfach viel angenehmer zu handhaben.Freundlicherweise gibt es schon ein CByteArray. Das erspart dir ein typedef.
Hier nochmal der Code mit CByteArray:#include "afxtempl.h" void find_used_dl(CByteArray& result) { for(int cnt = 0; cnt < 26; cnt++ ) { // Wiederhole für jeden Laufwerksbuchstaben CHAR dl = cnt + 65; // Damit kriegt man den Buchstaben int drivetype = GetDriveType( dl + ":" ); if (drivetype != 1) { result.Add(dl); result.Add(drivetype); cout << "find_used_dl(): Habe Laufwerk " << dl << ": gefunden! Typ: " << drivetype << "\n"; } } }Nun hat das aber einen kleinen Haken: Da es kein fertiges 2D-Array gibt, musst du ebn wissen, dass ungerade Elemente der Buchstabe sind und gerade der Typ.
Aufrufen kannst du das nun so:CByteArray arrLW; find_used_dl(arrLW); for (int i = 0; i < arrLW.GetSize(); i+=2) // Es wurde was gefunden { AfxMessageBox(arrLW.GetAt(i) + _T(" ist ") + arrLW.GetAt(i+1)); }Okay?

-
Ich persönlich würde versuchen ein struct hernehmen und nacher den Zeiger auf
dieses struct als Rückgabewert ausgeben.
-
Hallo,
danke für die Antwort.
@Bumsfallera:
Ich mache seit 3 Tagen C++. Vorher nur Perl/TK.
Was ist ein struct ???@estartu_de:
Ich rufe die Funktion in der main so auf:BYTE result[26][2]; BYTE *pointer; pointer = &result[26][2]; int used_dl_cnt = find_used_dl(pointer);Definiert habe ich das ganze in der .h Datei so:
int find_used_dl(BYTE*);Die Funktion (übernommen, ohne Änderung)
int find_used_dl(BYTE* result) { int result_cnt = 0; // Arrayzähler for(int cnt = 0; cnt < 26; cnt++ ) { // Wiederhole für jeden Laufwerksbuchstaben CHAR dl = cnt + 65; // Damit kriegt man den Buchstaben int drivetype = GetDriveType( dl + ":" ); if (drivetype != 1) { result[result_cnt][0] = dl; result[result_cnt][1] = drivetype; result_cnt++; cout << "find_used_dl(): Habe Laufwerk " << dl << ": gefunden! Typ: " << drivetype << "\n"; } } return result_cnt; }Das ergibt diese Fehler:
<PFAD>\device.cpp(47) : error C2109: subscript requires array or pointer type
<PFAD>\device.cpp(47) : error C2106: '=' : left operand must be l-value
<PFAD>\device.cpp(48) : error C2109: subscript requires array or pointer type
<PFAD>\device.cpp(48) : error C2106: '=' : left operand must be l-valueGibt es in C++ einen Datentyp der nach dem Thema "Halt die Klappe und mach einfach" arbeitet?
Ich versuche C++ zu lernen und jeder 2. Fehler lautet auf einen feherhaften Datentyp.
Es ist mir ehrlich gesagt §%"§%" egal, ob ein long oder double mehr Speicher addressieren, als ein int.
Mir kommt es so vor, dass es in C++ mindesten 100 Funktionen zum umwandlen von Datentypen geben muss...?
Kann doch gar nicht so schwer sein, oder?mfg
trequ
-
Hi, hilft dir das Edit oben weiter? Die Antworten haben sich genau überschnitten.
Eine allgemeine Sache: In C ist der Name eines Arrays der Zeiger auf das 0. Element. Das hilft bei dem Zeigerkram weiter.
Ein struct ist eine Zusammenfassung von mehreren Daten zu einem neuen Typ.
Sowas wie ein Punkt, der ist ja auch x und y Koordinate.Ja, es gibt viele Funktionen zum Umwandeln von Datentypen, auch wenn sich einige auch so verstehen. Schau dir mal Casts an.
-
ein struct kann man in etwa mit einer Tabelle vergleichen.
ihn kann man mit verschiedenen Möglichkeiten initialisieren (einfach mal nach
struct hier suchen).z.b.
struct person { CString name; CString vorname; int alter; int schuhgroesse; }pos[100]; //dieser kann 100 Datensätze verwalten //speichern wie folgt: pos[0].name = "Wurst"; pos[0].vorname = "Hans"; pos[0].alter =66; pos[0].schuhgroesse = 48;angaben ohne gewähr, da ich auf der arbeit sitze und das nur nebenher tippel.
vg!
-
Hallo,
ich habe das jetzt eingebaut:
<PFAD>\DriveLetter.cpp(44) : error C2297: '<' : illegal, right operand has type 'int (__thiscall CByteArray::*)(void) const'Es geht um die Zeile:
for (int i = 0; i < arrLW.GetSize; i+=2) // Es wurde was gefundenIch habe auch versucht es anders auszugeben:
cout << arrLW[0];
Keine Fehler beim erstellen, dafür extremer Programmabsturz...
#################
Ich glaube ich habe das mit dem struct verstanden, aber ich weiß nicht, wie ich das verwenden soll.
Mein Ziel ist es, eine Datei zu erstellen, welche alle Funktionen die ich mit den Laufwerken machen muss, beinhaltet.
Ich dachte, es reicht aus eine .cpp und eine .h Datei in das Projekt zu kopieren und die .h Datei zu includen.
Das struct müsste dann auch in die .cpp datei, oder?Noch was zu Thema struct:
http://msdn.microsoft.com/library/en-us/devio/base/ntms_driveinformation.asp
Ist ein struct, welches in der Ntmsapi.h enthalten ist.
Ich habe daher dies probiert:#include "Ntmsapi.h" ... cout << _NTMS_DRIVEINFORMATION.Number;Dies liefert jedoch:
<PFAD>\1.cpp(40) : error C2275: '_NTMS_DRIVEINFORMATIONA' : illegal use of this type as an expression
<PFAD>\include\ntmsapi.h(328) : see declaration of '_NTMS_DRIVEINFORMATIONA'
Warum?mfg
trequ
-
mit struct kannste datensätze einfacher handlen, weils einfach übersichtlicher
ist und du nur 1 zeiger brauchst, um auf die entsprechenden datensätze zuzu-
greifen (*pos).
-
Oh Gott, da fehlt die Klammer!
Sorry!!!!!!
for (int i = 0; i < arrLW.GetSize(); i+=2) // Es wurde was gefunden
-
Hallo,
....logisch, die Klammern sind hilfreich. (Hätte ich auch selber draufkommen können)
Ich habe den Code jetzt umgeschrieben:
int find_used_dl(CByteArray& result) { int result_cnt = 0; // Arrayzähler int drivetype; // Laufwerkstyp DWORD drives; // Verwendete Laufwerke char drive[8] = "A:\\"; // Laufwerksbuchstabe drives = GetLogicalDrives(); for(int i=0; i<26; i++) { if(drives) { drive[0] = i + 'A'; drivetype = GetDriveType(drive); if(drivetype != 1){ result_cnt++; //result.Add(drive); <<-- Ergibt Fehler result.Add(drivetype); cout << "Drive: " << drive << " Typ: " << drivetype << "\n"; } } } return result_cnt; }Es werden bei mir 12 Laufwerke gefunden (passt zum Arbeitsplatz) auch Laufwerksbuchstabe und Typ werden richtig ausgegeben.
Allerdings funktioniert das mit dem Array immer noch nicht.
Die Zeile
result.Add(drive);
erzeugt
<PFAD>\device.cpp(53) : error C2664: 'Add' : cannot convert parameter 1 from 'char [8]' to 'unsigned char'
This conversion requires a reinterpret_cast, a C-style cast or function-style castIch habe die Zeile auskommentiert und den Wert von GetSize in der main ausgelesen. 12. Passt. Auch der return (int) hat den Wert 12.
Allerdings stehen nur Herzchen in dem Array:CByteArray arrLW; int used_dl_cnt = find_used_dl(arrLW); for (int i = 0; i < arrLW.GetSize(); i+=2) // Es wurde was gefunden { cout << arrLW.GetAt(i) << arrLW.GetAt(i+1) << "\n"; } cout << arrLW.GetSize() << "\n---\n";erzeugt:
☻










12
---
Ich werde mit dem ganzen Datentypen von C++ noch Wahnsinnig.
Gibt es keine Datentyp so nach dem Motto "Hat die Klappe und mach einfach"
mfg
trequ
-
gibt es gott sei dank NICHT.
entweder würde dann nur unnötig speicher verpulvert oder es gäbe Overflows am
laufenden band
da musst du wohl durch *hehe*
-
Der Mist hat mir grade noch gefehlt...
In Perl musste ich mir da nie große Gedanken drum machen...
Warscheinlich macht sich C++ im Lebenslauf so gut, wie alle sagen.
Aber warum ich von char nicht nach char[8] kann, verstehe ich nicht.
Was hat die [8] da zu bedeuten?mfg
trequ
-
du meinst wohl
char variablenname[8];das wäre ein array vom typ char in dem du 8 char-werte speichern kannst.
und: ja, c++ im lebenslauf isn echter kaufgrund
-
Hallo,
demnach ist char also ein einzelner Buchstabe...
-> Wer hat sich denn den Mist ausgedacht? Dann kann man ja gleich CString nehmen...Sorry, aber das löst alles das Problem nicht...
Da muss ja irgend etwas falsch umgewandelt werden, die Sonderzeichen kommen ja irgendwo her...mfg
trequ
-
geht nicht:
char drive[8] = "A:\\";geht:
char drive[] = "A:\\";
-
versuch mal nen array im CString-Format, in dem du die laufwerksbezeichnungn
wie "A:\" zum schluss speicherst und ein char für den aktuellen laufwerks-
buchstaben zu ermitteln und dann in die CString incl. ":\" zu speichern
-
Hi,
wenn ich drive[] schreibe, kriege ich einen Fehler
"Unable to convert char[4] to char" oder so ähnlich.Wenn ich statt CByteArray ein CString Array verwende, kriege ich lauter Zahlen anstelle von Sonderzeichen.
--> Bitte bedenken: Ich mache C++ erst seit 3 oder 4 Tagen...

Ich verstehe es einfach nicht:
In der Praxis wird man doch öfter mal ein Array von einer Funktion zurückgeben müssen.Das kann doch nicht so schwer sein...

mfg
trequ
-
versuch mal einen CString-Array als membervariable einzufügen und verwende den.
du hast bei einer funktion nie einen array als rückgabewert, höchstens nen
zeiger auf diesen array.
für den laufwerksbuchstaben brauchst du keinen char-array, da nur 1 Buchstabe.
verwende ne einfache char-variable als zwischenspeicher um die dann in den
CString zu formatieren und das ":\" dranzuhängen
-
oder vielleicht hilft dir das weiter:
#include <string.h> #include <stdio.h> void main( void ) { char string[80]; strcpy( string, "Hello world from " ); strcat( string, "strcpy " ); strcat( string, "and " ); strcat( string, "strcat!" ); printf( "String = %s\n", string ); }Output:
String = Hello world from strcpy and strcat!
-
Hallo!
In deinem Beitrag von 16:37:24 19.08.2004 habe ich den Fehler gefunden:
Du hast den Variablentyp geändert. Ich hatte eine Variable dl vom Typ CHAR (den man problemlos in ein BYTE stecken kann) und du hast daraus einen char[8] gemacht. Das geht nicht. Diese Typen vertragen sich nicht.Warum? Ein BYTE ist eine 8 Bit große Speicherstelle. Ein char auch. Ein char[8] sind 8 chars. Nun überleg mal, warum 8 Teile nicht auf den Platz von einem Teil passen.

Als Abhilfe kannst du drive[0] zuweisen. Das ist ja der Laufwerksbuchstabe und nur EIN Buchstabe und nicht gleich mehrere.
Du hast dir eine ziemlich anspruchsvolle Aufgabe gesucht, um dich mit C++ anzufreunden. Kann es sein, dass du den Fehler machst, dich wegen deiner schon vorhandenen Programmierkenntnisse zu übernehmen?
Du magst Pearl gut können, vielleicht sogar sehr gut - aber das heißt nicht, dass du andere Sprachen auch gut kannst. Fang langsam an oder ärgere dich nicht so sehr darüber, dass du kaum voran kommst. (Auch wenn dir ein Chef im Nacken sitzt - davon wirst du auch nicht schneller lernen.)PS: Ohne in Büchern oder der MSDN zu lesen wirst du nicht viel Freude haben, denn es wird sich kaum jemand finden, der die Geduld hat, dir alles haarklein vorzukauen.
C ist nicht einfach, C++ ist noch viel mehr als C und die MFC ist nochmal mehr zu lernen - erkennst du nun, dass du es nicht in einer Woche verstehen wirst?
Schau ruhig mal nach, was ein Begriff bedeutet, der dir um die Ohren geschlagen wird. Wenn du nicht weißt, was eine Referenz ist, dann such die Definition, und frag nach, wenn du sie nicht verstehst. Oder arbeite dich durch ein Buch. Es gibt so viele...
-
Hallo!
ich habe eine Frage .
Wie kann ich die Funktion zum anzeigen von Laufwerken so umformen, damit man die in ein ComboBox einbauen kann?