Обнаружение 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 оценивает значение предопределенной константы во время компиляции.

Другие вопросы по тегам