Обнаружение Windows или Linux в C, 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;
}
5 ответов
Обычно это делается так (более или менее):
#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.
Вы должны использовать то же самое #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
}
Здесь я вижу много разных решений, что вызывает у меня неудобство... Что если они работают на Linux, но не на Windows или на Windows, но не на Linux? Что если они работают только на некоторых компиляторах? И т.п.
Поэтому я нашел эту ссылку, которая мне нравится: 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
Вы путаете переменные (которые существуют во время выполнения) с символами препроцессора (которые существуют только во время компиляции).
После того, как вы делаете что-то вроде #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
}
Ваша основная стратегия глубоко испорчена.
Используя if в основной функции, вы выбираете, какой фрагмент кода запускать в режиме RUNTIME. Таким образом, даже если компиляция для unix, компилятору все равно придется создавать код для окон (что он не может сделать, поскольку файлы заголовков не включены) и наоборот.
Вместо этого вам требуется #if, который будет оцениваться во время компиляции и не будет пытаться скомпилировать нерелевантный код.
По сути, вам нужно понимать, что if оценивает ваше определение как выражение во время выполнения, тогда как #if оценивает значение предопределенной константы во время компиляции.