Обнаружение Windows или Linux в C, С++
Я пишу кросс-платформенную программу. Я хочу, чтобы эта одна программа работала под Windows и Linux, поэтому у меня есть два разных сегмента кода для двух платформ. Если ОС - Windows, я хочу, чтобы первый сегмент кода выполнялся; если это Linux, то я хочу, чтобы второй сегмент кода работал.
Итак, я написал следующий код, но он получает ошибку при построении как в Windows, так и в Linux. Что делать, чтобы решить эту проблему?
#ifdef __unix__ /* __unix__ is usually defined by compilers targeting Unix systems */
#define OS_Windows 0
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#elif defined(_WIN32) || defined(WIN32) /* _Win32 is usually defined by compilers targeting 32 or 64 bit Windows systems */
#define OS_Windows 1
#include <windows.h>
#include <stdio.h>
#include <tchar.h>
#define DIV 1048576
#define WIDTH 7
#endif
int main(int argc, char *argv[])
{
if(OS_Windows)
{
MEMORYSTATUSEX statex;
statex.dwLength = sizeof (statex);
GlobalMemoryStatusEx (&statex);
_tprintf (TEXT("There is %*ld %% of memory in use.\n"),
WIDTH, statex.dwMemoryLoad);
}
else if(!OS_Windows) // if OS is unix
{
char cmd[30];
int flag = 0;
FILE *fp;
char line[130];
int memTotal, memFree, memUsed;
flag=0;
memcpy (cmd,"\0",30);
sprintf(cmd,"free -t -m|grep Total");
fp = popen(cmd, "r");
while ( fgets( line, sizeof line, fp))
{
flag++;
sscanf(line,"%*s %d %d %d",&TotalMem, &TotalUsed, &TotalFree);
}
pclose(fp);
if(flag)
printf("TotalMem:%d -- TotalUsed:%d -- TotalFree:%d\n",TotalMem,TotalUsed,TotalFree);
else
printf("not found\n");
}
return 0;
}
Ответы
Ответ 1
Обычно это делается (более или менее):
#ifdef _WIN32
#include <windows.h>
#include <stdio.h>
#include <tchar.h>
#define DIV 1048576
#define WIDTH 7
#endif
#ifdef linux
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#endif
int main(int argc, char *argv[])
{
#ifdef _WIN32
MEMORYSTATUSEX statex;
statex.dwLength = sizeof (statex);
GlobalMemoryStatusEx (&statex);
_tprintf (TEXT("There is %*ld %% of memory in use.\n"),
WIDTH, statex.dwMemoryLoad);
#endif
#ifdef linux
char cmd[30];
int flag = 0;
FILE *fp;
char line[130];
int TotalMem, TotalFree, TotalUsed;
flag=0;
memcpy (cmd,"\0",30);
sprintf(cmd,"free -t -m|grep Total");
fp = popen(cmd, "r");
while ( fgets( line, sizeof line, fp))
{
flag++;
sscanf(line,"%*s %d %d %d",&TotalMem, &TotalUsed, &TotalFree);
}
pclose(fp);
if(flag)
printf("TotalMem:%d -- TotalUsed:%d -- TotalFree:%d\n",TotalMem,TotalUsed,TotalFree);
else
printf("not found\n");
#endif
return 0;
}
Таким образом, только код для linux будет скомпилирован на платформе linux, и только Windows-код будет скомпилирован на платформе Windows.
Ответ 2
В коде следует использовать те же #ifdef
вместо if(OS_Windows)
логики:
#ifdef __unix__
...
#elif defined(_WIN32) || defined(WIN32)
#define OS_Windows
#endif
int main(int argc, char *argv[])
{
#ifdef OS_Windows
/* Windows code */
#else
/* GNU/Linux code */
#endif
}
Ответ 3
Вы смешиваете переменные (которые существуют во время выполнения) с символами препроцессора (которые существуют только во время компиляции).
После того, как вы сделаете что-то вроде #define OS_Windows 1
, вы не можете использовать символ OS_Windows в качестве переменной и поместить его внутри if()
s... Я имею в виду, вы можете, но он будет расширен во время компиляции до if (1)
.
Для кросс-платформенного проекта вам нужно использовать #if
или #ifdef
, чтобы убедиться, что компилятор выбирает правильную часть кода для данной ОС и компилирует только что.
Что-то вроде:
void openWindow() {
#if OS_Windows
// windows-specific code goes here
#else
// linux-specific code
#endif
}
Ответ 4
Здесь я вижу много разных решений, что делает меня неудобным... Что делать, если они работают на Linux, но не на Windows или Windows, а на Linux? Что делать, если они работают только на некоторых компиляторах? Etc.
Итак, я нашел эту ссылку, которая мне нравится: http://nadeausoftware.com/articles/2012/01/c_c_tip_how_use_compiler_predefined_macros_detect_operating_system
Похоже, что это лучше всего (используя #ifdef, #endif и т.д.):
-
_WIN32
для Windows 32 бит
-
_WIN64
для Windows 64 бит
-
__unix__
для Unix
Ответ 5
Ваша основная стратегия глубоко испорчена.
Используя функцию if в основной функции, вы выбираете, какой фрагмент кода запускается в RUNTIME.
Таким образом, даже если компиляция для unix компилятору по-прежнему приходится создавать код для окон (который он не выполняет, поскольку файлы заголовков не включены) и наоборот.
Вместо этого вы должны указать #if, который будет оцениваться во время компиляции и не будет пытаться скомпилировать нерелевантный код.
Таким образом, в сущности, вам нужно понять, что если вы оцениваете свое определение как выражение во время выполнения, тогда как #if оценивает значение предопределенной константы во время компиляции.