![]() |
[C] Zeichenfolge aus Funktion zurückgeben
Folgender Code:
Code:
Leider gelingt es mir nicht eine ganz normale Zeichenfolge beliebiger Länmge aus der Funktion zurückzugeben, so dass ich sie mit printf ausgeben kann.
#include <stdio.h>
#include <windows.h> TCHAR* SysErrorMessage(int ErrorCode) { TCHAR szBuf[80]; LPVOID lpMsgBuf; DWORD dw = GetLastError(); FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, dw, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL ); wsprintf(szBuf, (LPTSTR)lpMsgBuf); LocalFree(lpMsgBuf); return szBuf; } int main(int argc, char* argv[]) { printf(SysErrorMessage(5)); return 0; } Zusatzaufgabe: Wie mache ich das ganze Unicode sicher? |
Re: [C] Zeichenfolge aus Funktion zurückgeben
Benutzt du dort den GNU-C Compiler?
Mh, ohne jetzt eine genaue Ahnung zu haben, würde ich versuchen einen pChar zurückzugeben. Grad mal in die Hilfe zu FormatMessage geguckt: Zitat:
|
Re: [C] Zeichenfolge aus Funktion zurückgeben
Code:
- übergebenen Wert aus "ErrorCode" wird jetzt zur Umwandlung genommen
#include <stdio.h>
#include <windows.h> TCHAR* SysErrorMessage(int ErrorCode) { TCHAR szBuf[80]; LPVOID lpMsgBuf; // DWORD dw = GetLastError(); DWORD dw = ErrorCode; FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, dw, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR)&lpMsgBuf, 0, NULL ); wsprintf(szBuf, (LPTSTR)lpMsgBuf); LocalFree(lpMsgBuf); return szBuf; } int main(int argc, char* argv[]) { printf("%s", SysErrorMessage(5)); return 0; } - Ausgabe geändert in "printf("%s", SysErrorMessage(5));" Ich würde einen Char-Array als Parameter übergeben, in den du schreibst, oder du holst Speicher innerhalb der Funktion und gibst ihn nach dem Aufrufen wieder frei. :) |
Re: [C] Zeichenfolge aus Funktion zurückgeben
Zitat:
Deine Version kompiliert jetzt zwar, aber es wird nur Datenmüll ausgegeben. Später will ich die Funktion SysErrorMessage auch bei einer Messagebox oder so benutzen, da geht das ja dann nicht mehr mit printf("%s", ...);. |
Re: [C] Zeichenfolge aus Funktion zurückgeben
Ganz grober Schnitzer: szBuf liegt auf dem Stack, ist also nach dem Aufruf der Funktion eigentlich nicht mehr gültig. Wenn der Inhalt noch drin steht, dann hast du Glück. Ein erster Bugfix wäre, static TCHAR szBuf[80]; zu benutzen.
Allerdings ist es in C üblich, den Ergebnisparameter zusammen mit der maximalen Länge zu übergeben, wie ste_ett schon schrieb. |
Re: [C] Zeichenfolge aus Funktion zurückgeben
Also so was:
Code:
Nur bin ich in C noch nicht so tritt fest, wie müsste denn dann die Implementation und der Aufruf aussehen? Und ambesten noch Unicode fähig. Wir wollen ja gleich up to date sein. ;)
void SysErrorMessage(int ErrorCode, char* Buffer, int lenBuffer)
|
Re: [C] Zeichenfolge aus Funktion zurückgeben
Code:
z.B. so :)
#include <stdio.h>
#include <windows.h> DWORD SysErrorMessage(const int ErrorCode, wchar_t* szBuf, const DWORD ArrayLength) { DWORD result = 0; LPVOID lpMsgBuf = NULL; DWORD dw = ErrorCode; result = FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, dw, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPWSTR)&lpMsgBuf, 0, NULL); if (result <= ArrayLength) { wsprintf(szBuf, (LPWSTR)lpMsgBuf); } else { result = (DWORD)-1; // ErrCode } LocalFree(lpMsgBuf); return result; } int main(int argc, char* argv[]) { TCHAR szBuf[80]; ZeroMemory(szBuf, sizeof(szBuf)); if (SysErrorMessage(5, szBuf, sizeof(szBuf) / 2)) { wprintf(L"%s", szBuf); } return 0; } |
Re: [C] Zeichenfolge aus Funktion zurückgeben
Etwa so:
Code:
// Nachtrag: noch erweitert
TCHAR *
SysErrorMessage (DWORD dwErrorCode, TCHAR *pszBuf, int nLenBuf) { FormatMessage( FORMAT_MESSAGE_FROM_SYSTEM | (pszBuf ? FORMAT_MESSAGE_ALLOCATE_BUFFER : 0), NULL, dwErrorCode ? dwErrorCode : GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), pszBuf ? pszBuf : (LPTSTR)&pszBuf, pszBuf ? nLenBuf : 0, NULL); return pszBuf; } |
Re: [C] Zeichenfolge aus Funktion zurückgeben
Ah, ja die Version von ste_ett funktioniert und ich habe sie verstanden. Aber wäre es nicht sinnvoll die benötigte Länge des Buffers zurückzugeben, so dass man im Bedarfsfall entsprechend mehr Speicher reservieren kann? aslo:
Code:
Das ist jetzt Delphi Pseudocode.
len = SysErrorMessage(5, NULL, 0);
GetMem(szBuff, len) SysErrorMessage(5,szBuf, len); FreeMem(szBuf); Und wie kann ich den Code in eine eigene Quellcodedatei auslagern? Ich habe ihn in eine eigene Datei kopiert MpuTools.cpp und eine entsprechende headerdatei angelegt: MpuTools.h, die ich dann mit include eingebunden habe:
Code:
Das mag der Compiler jedoch nicht:
#include <stdio.h>
#include <windows.h> #include "MpuTools.h" int main(int argc, char* argv[]) { TCHAR szBuf[80]; ZeroMemory(szBuf, sizeof(szBuf)); if (SysErrorMessage(5, szBuf, sizeof(szBuf)) >0) { wprintf(L"%s", szBuf); MessageBoxW(0, szBuf, L"Test", 0); } return 0; } Zitat:
Zitat:
|
Re: [C] Zeichenfolge aus Funktion zurückgeben
OK, das Problem ist gelöst: Ich musste noch die dateien zum Projekt hinzufügen. Ich dachte das passiert irgendwie automatisch, wenn ich die entsprechende Header-Datei einbinde. Damit wäre alles geklärt. Besten Dank für eure Hilfe.
|
Re: [C] Zeichenfolge aus Funktion zurückgeben
Zitat:
Code:
Diese Routine kannst du auf variable Art aufrufen, bei C++ könntest du sogar noch Defaultwerte für die Parameter angeben.
TCHAR *
SysErrorMessage (DWORD dwErrorCode, TCHAR *pszBuf, int nLenBuf) { FormatMessage( FORMAT_MESSAGE_FROM_SYSTEM | (pszBuf ? FORMAT_MESSAGE_ALLOCATE_BUFFER : 0), NULL, dwErrorCode ? dwErrorCode : GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), pszBuf ? pszBuf : (LPTSTR)&pszBuf, pszBuf ? nLenBuf : 0, NULL); return pszBuf; }
Code:
#define countof(X) (sizeof(X) / sizeof(X[0]))
int main (int argc, char **argv) { // Statischer Puffer mit fester Länge, Fehlermeldung 5 TCHAR szBuf[80]; printf(_T("%s\n"), SysErrorMessage(5, szBuf, countof(szBuf))); // Dynamischer String, aktuelle Fehlermeldung LPTSTR pszBuf; pszBuf = SysErrorMessage(0, NULL, 0); printf(_T("%s\n"), pszBuf); LocalFree(pszBuf); return 0; } |
Re: [C] Zeichenfolge aus Funktion zurückgeben
Zitat:
Er meint jetzt aber dass _T nicht gefunden wurde: Zitat:
|
Re: [C] Zeichenfolge aus Funktion zurückgeben
Habs gelöst, man muss #include <tchar.h> inkludieren. Nur bleibt meine Messagebox leer und bei printf kommt:
Zitat:
Code:
So:
#include <stdio.h>
#include <windows.h> #include <tchar.h> #include "MpuTools.h" int main(int argc, char* argv[]) { // Statischer Puffer mit fester Länge, Fehlermeldung 5 TCHAR szBuf[80]; //printf(_T("%s\n"), SysErrorMessage(5, szBuf, countof(szBuf))); // Dynamischer String, aktuelle Fehlermeldung LPTSTR pszBuf; pszBuf = SysErrorMessage(0, NULL, 0); //printf(_T("%s\n"), pszBuf); LocalFree(pszBuf); MessageBoxW(0, SysErrorMessage(5, NULL, 0), L"Test", 0); return 0; }
Code:
Ist die erste Messagebox leer und die zweite zeigt nur viele quadrate an. :?
int main(int argc, char* argv[])
{ // Dynamischer String, aktuelle Fehlermeldung LPTSTR pszBuf; pszBuf = SysErrorMessage(0, NULL, 0); MessageBoxW(0, pszBuf, L"Test", 0); //printf(_T("%s\n"), pszBuf); LocalFree(pszBuf); // Statischer Puffer mit fester Länge, Fehlermeldung 5 TCHAR szBuf[80]; MessageBoxW(0, SysErrorMessage(5, szBuf, countof(szBuf)), L"Test", 0); return 0; } |
Re: [C] Zeichenfolge aus Funktion zurückgeben
Sorry Luckie.
1. Ich hätte für printf natürlich _tprintf nehmen müssen. Die ganzen _t-Funktionen sind auch in tchar.h definiert. 2. Das MessageBoxW sollte eigentlich nicht nötig sein. Hast du UNICODE definiert? Übersetzt du mit dem GCC oder MSC? // Nachtrag Hab einen Dreher dringehabt :oops: (kommt davon wenn man's nicht ausprobiert)
Code:
TCHAR *
SysErrorMessage (DWORD dwErrorCode, TCHAR *pszBuf, int nLenBuf) { FormatMessage( FORMAT_MESSAGE_FROM_SYSTEM | (pszBuf ? 0 : FORMAT_MESSAGE_ALLOCATE_BUFFER), NULL, dwErrorCode ? dwErrorCode : GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), pszBuf ? pszBuf : (LPTSTR)&pszBuf, pszBuf ? nLenBuf : 0, NULL); return pszBuf; } |
Re: [C] Zeichenfolge aus Funktion zurückgeben
Ah, das erklärt einiges. ;) Besten Dank.
|
Re: [C] Zeichenfolge aus Funktion zurückgeben
Die Frage ist noch als offen markiert.
Hinzufügen möchte ich auch, daß der gute Stil erfordert, daß man absolut niemals (s)printf, ebenso wie strcat, strcpy usw. benutzen sollte (auch ich halte mich nicht immer dran ...). Warum sollte bekannt sein, wenn man ab und zu die Meldungen über Verwundbarkeiten in Software verfolgt. Alle diese Funktionen achten nicht auf die Länge des Puffers - daher kommt es gehäuft beim Einsatz dieser Funktionen zu Sicherheitslücken. Es gibt für die meisten dieser Funktionen eine Version mit Längenüberprüfung. Von der Typunsicherheit habe ich noch nichtmal geredet. Unter C++ sind sowieso die Streams vorzuziehen, da du die Typsicherheit schon während des Kompilierens gewährleistest. Ansonsten machst du nichts anderes als ein C-Programm mit einem C++-Compiler zu kompilieren ... :zwinker: |
Alle Zeitangaben in WEZ +1. Es ist jetzt 17:05 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz