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 사용 등).
_WIN32
Windows 32비트용_WIN64
Windows 64비트용__unix__
UNIX의 경우
변수(런타임에 존재하는 것)와 프리프로세서 기호(컴파일 중에만 존재하는 것)를 혼동하고 있습니다.
그런 걸 하고 나면#define OS_Windows 1
OS_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
'programing' 카테고리의 다른 글
JPA OneToMany가 하위 항목을 삭제하지 않음 (0) | 2022.08.10 |
---|---|
Java로 정렬된 컬렉션 (0) | 2022.08.10 |
Java 8의 map() 메서드와 platMap() 메서드의 차이점은 무엇입니까? (0) | 2022.08.10 |
0x 표기법을 사용한 숫자는 무엇을 의미합니까? (0) | 2022.08.10 |
라우터에서 ASync/Awit가 예상대로 작동하지 않습니다.교도소에 있는 각 간수들 전에? (0) | 2022.08.10 |