Подтвердить что ты не робот

Обнаружение 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;
}
4b9b3361

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