Comment convertir de _TCHAR * en char * lorsque j'utilise des arguments de longueur variable C ++?

StackOverflow https://stackoverflow.com/questions/1613217

  •  06-07-2019
  •  | 
  •  

Question

Nous devons passer une chaîne de format _TCHAR * et un certain nombre de chaînes char * dans une fonction avec des arguments de longueur variable:

inline void FooBar(const _TCHAR *szFmt, const char *cArgs, ...) {
  //...
}

On peut donc l'appeler ainsi:

char *foo = "foo";
char *bar = "bar";
LogToFileA(_T("Test %s %s"), foo, bar);

Évidemment, une solution simple serait d'utiliser _TCHAR au lieu de char, mais nous n'avons malheureusement pas ce luxe.

Nous devons utiliser cela avec va_start, etc. pour pouvoir formater une chaîne:

va_list args;
_TCHAR szBuf[BUFFER_MED_SIZE];

va_start(args, cArgs);
_vstprintf_s(szBuf, BUFFER_MED_SIZE, szFmt, args);
va_end(args);

Malheureusement, nous ne pouvons pas l'utiliser car cela nous donne cette erreur:

Unhandled exception at 0x6a0d7f4f (msvcr90d.dll) in foobar.exe:
0xC0000005: Access violation reading location 0x2d86fead.

Je pense que nous devons convertir notre caractère * en _TCHAR * - mais comment?

Était-ce utile?

La solution

Utilisez% hs ou% hS au lieu de% s. Cela forcera les paramètres à être interprétés comme char * dans les versions Ansi et Unicode des fonctions de style printf (), c'est-à-dire:

inline void LogToFile(const _TCHAR *szFmt, ...)
{  
  va_list args;
  TCHAR szBuf[BUFFER_MED_SIZE];

  va_start(args, szFmt);
  _vstprintf_s(szBuf, BUFFER_MED_SIZE, szFmt, args);
  va_end(args);
}  

{
  char *foo = "foo"; 
  char *bar = "bar"; 
  LogToFile(_T("Test %hs %hs"), foo, bar); 
}

Autres conseils

Cela ressemble généralement à ce qui suit:

char *foo = "foo";
char *bar = "bar";
#ifdef UNICODE
LogToFileW( L"Test %S %S", foo, bar); // big S
#else
LogToFileA( "Test %s %s", foo, bar);
#endif

Votre question n’est pas tout à fait claire. Comment votre fonction est implémentée et comment l’utilisez-vous?

Voici ma solution - j'apprécie les suggestions d'amélioration!

inline void FooBar(const _TCHAR *szFmt, const char *cArgs, ...) {

    va_list args;
    _TCHAR szBuf[BUFFER_MED_SIZE];

    // Count the number of arguments in the format string.
    const _TCHAR *at = _tcschr(szFmt, '%');
    int argCount = 0;
    while(at) {
        argCount++;
        at = _tcschr(at + 1, '%');
    }

    CA2W *ca2wArr[100];
    LPWSTR szArgs[100];
    va_start(args, cArgs);
    for (int i = 1; i < argCount + 1; i++) {
        CA2W *ca2w = new CA2W(cArgs);
        szArgs[i] = ca2w->m_psz;
        ca2wArr[i] = ca2w;
        cArgs = va_arg(args, const char *);
    }
    va_end(args);

    // Use the new array we just created (skips over first element).
    va_start(args, szArgs[0]);
    _vstprintf_s(szBuf, BUFFER_MED_SIZE, szFmt, args);
    va_end(args);

    // Free up memory used by CA2W objects.
    for (int i = 1; i < argCount + 1; i++) {
        delete ca2wArr[i];
    }

    // ... snip ... - code that uses szBuf
}

c’est quelque chose que j’avais utilisé auparavant pour convertir un TCHAR en caractère, espérons que cela aide, bien que je ne cherche pas vraiment à l’optimisation, ce n’est donc pas le moyen le plus rapide .. mais cela a fonctionné!

    TCHAR tmp[255];
::GetWindowText(hwnd, tmp, 255);
std::wstring s = tmp;

//convert from wchar to char
const wchar_t* wstr = s.c_str();
size_t wlen = wcslen(wstr) + 1;
char newchar[100];
size_t convertedChars = 0;
wcstombs_s(&convertedChars, newchar, wlen, wstr, _TRUNCATE);
Licencié sous: CC-BY-SA avec attribution
Non affilié à StackOverflow
scroll top