programing

C, C++에서 Windows 또는 Linux 검출

bestcode 2022. 8. 10. 22:30
반응형

C, C++에서 Windows 또는 Linux 검출

크로스 플랫폼 프로그램을 쓰고 있습니다.이 하나의 프로그램을 Windows와 Linux 모두에서 실행하고 싶기 때문에 2개의 플랫폼에 대해 2개의 다른 코드 세그먼트를 가지고 있습니다.OS가 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;
}

일반적으로 다음과 같이 처리됩니다(거의).

#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에서는 동작하지 않거나 Linux에서는 동작하지 않는 경우는 어떻게 됩니까?일부 컴파일러에서만 작동한다면?기타.

https://web.archive.org/web/20191012035921/http://nadeausoftware.com/articles/2012/01/c_c_tip_how_use_compiler_predefined_macros_detect_operating_system 라는 링크를 찾았습니다.

이것이 최적인 것 같습니다(#ifdef, #endif 사용 등).

  • _WIN32Windows 32비트용
  • _WIN64Windows 64비트용
  • __unix__UNIX의 경우

변수(런타임에 존재하는 것)와 프리프로세서 기호(컴파일 중에만 존재하는 것)를 혼동하고 있습니다.

그런 걸 하고 나면#define OS_Windows 1OS_Windows 기호를 변수로 사용할 수 없습니다.if()제 말은... 할 수는 있지만 컴파일하는 동안 확장될 거예요if (1).

크로스 플랫폼프로젝트의 경우 컴파일러가 특정 OS에 대해 올바른 코드 부분을 선택하여 컴파일하는 것을 또는 를 사용해야 합니다.

예를 들어 다음과 같습니다.

void openWindow() {
#if OS_Windows
    // windows-specific code goes here
#else
    // linux-specific code
#endif
}

당신의 기본 전략은 심각한 결함이 있습니다.

메인 함수에서 if를 사용하면 런타임에 실행할 코드 청크를 선택할 수 있습니다.따라서 unix용 컴파일러는 Windows용 코드를 작성해야 합니다(헤더 파일이 포함되어 있지 않기 때문에 작성하지 못하고 있습니다).

대신 컴파일 시 평가되며 관련 없는 코드를 컴파일하지 않는 #if를 요구했습니다.

따라서 기본적으로는 실행 시 식으로서 정의되는 반면 #if는 컴파일 시 사전 정의된 상수 값을 평가한다는 점을 이해해야 합니다.

언급URL : https://stackoverflow.com/questions/8666378/detect-windows-or-linux-in-c-c

반응형