C 및 C++에 대한 Microsoft 확장
업데이트: 2007년 11월
다음은 ANSI C 및 ANSI C++ 표준에 대한 Visual C++ 확장입니다.
키워드
Microsoft에서는 여러 가지 추가 키워드를 사용하여 C++ 언어를 확장합니다. 전체 목록을 보려면 C++ 언어 참조에서 C++ Keywords를 참조하십시오. 선행하는 두 개의 밑줄을 가진 키워드는 Microsoft 확장입니다.
static const 정수 계열(또는 열거형) 멤버의 Out-of-Class 정의
표준(/Za) 옵션에서는 데이터 멤버에 대해 out-of-class를 정의해야 합니다. 예를 들면 다음과 같습니다.
class CMyClass {
static const int max = 5;
int m_array[max];
}
...
const int CMyClass::max; // out of class definition
/Ze를 사용하면 static const 정수 계열 및 const 열거형 데이터 멤버에 대해 선택적으로 out-of-class를 정의할 수도 있고 정의하지 않을 수도 있습니다. static 및 const인 정수 계열과 열거형만이 클래스 내부에 초기 값을 가질 수 있으며, 초기화 식은 const 식이어야 합니다.
out-of-class가 정의될 때(out-of-class가 헤더 파일에 정의되고 헤더 파일이 다중 소스 파일에 포함될 때) 오류가 발생하지 않게 하려면, selectany를 사용해야 합니다. 예를 들면 다음과 같습니다.
__declspec(selectany) const int CMyClass::max = 5;
캐스트
컴파일러는 다음과 같은 두 개의 비ANSI 캐스트를 지원합니다.
l-values를 산출하기 위해 비ANSI 캐스트 사용
char *p; (( int * ) p )++;
앞의 예제를 다음과 같이 ANSI C 표준에 맞게 다시 작성할 수 있습니다.
p = ( char * )(( int * )p + 1 );
함수 포인터를 데이터 포인터로 비ANSI 캐스팅
int ( * pfunc ) (); int *pdata; pdata = ( int * ) pfunc;
ANSI 규격을 유지하면서 동일한 캐스트를 수행하려면, 함수 포인터를 데이터 포인터로 캐스팅하기 전에 int로 캐스팅해야 합니다.
pdata = ( int * ) (int) pfunc;
가변 길이 인수 목록
컴파일러는 다양한 수의 인수가 지정되는 함수 선언자 사용을 지원하며, 대신 이 뒤에는 형식을 제공하는 함수 정의가 옵니다.
void myfunc( int x, ... );
void myfunc( int x, char * c )
{ }
한 줄로 된 주석
C 컴파일러는 앞에 두 개의 슬래시(//) 문자가 있는 한 줄로 된 주석을 지원합니다.
// This is a single-line comment.
범위
C 컴파일러는 다음과 같은 범위 관련 기능을 지원합니다.
extern 항목을 static으로 재정의
extern int clip(); static int clip() {}
같은 범위 내에서 typedef 재정의 사용
typedef int INT; typedef int INT;
파일 범위의 함수 선언자
void func1() { extern int func2( double ); } int main( void ) { func2( 4 ); // /Ze passes 4 as type double } // /Za passes 4 as type int
비상수 식으로 초기화된 블록 범위의 변수 사용
int clip( int ); int bar( int ); int main( void ) { int array[2] = { clip( 2 ), bar( 4 ) }; } int clip( int x ) { return x; } int bar( int x ) { return x; }
데이터 선언 및 정의
C 컴파일러는 다음과 같은 데이터 선언 및 정의 기능을 지원합니다.
이니셜라이저에서 문자 및 문자열 상수 혼합 사용
char arr[5] = {'a', 'b', "cde"};
unsigned int 또는 signed int가 아닌 기본 형식의 비트 필드
저장소 클래스나 형식 중 하나가 없는 선언자
x; int main( void ) { x = 1; }
구조체와 공용 구조체의 마지막 필드로 크기를 지정하지 않은 배열 사용
struct zero { char *c; int zarray[]; };
명명되지 않은(익명) 구조체
struct { int i; char *s; };
명명되지 않은(익명) 공용 구조체
union { int i; float fl; };
명명되지 않은 멤버
struct s { unsigned int flag : 1; unsigned int : 31; }
내장 부동 소수점 함수
/Oi를 지정한 경우 컴파일러는 atan, atan2, cos, exp, log, log10, sin, sqrt 및 tan 함수 END x86 Specific의 인라인 생성 **x86 Specific >**을 지원합니다. C의 경우, ANSI 규칙은 내장 함수가 errno 변수를 설정하지 않으므로 이러한 내장 함수가 사용될 때는 적용되지 않습니다.
비 Const 포인터 매개 변수를 Const 포인터 매개 변수 참조를 원하는 함수에 전달
이는 C++에 대한 확장입니다. 다음 코드는 /Ze를 사용하여 컴파일합니다.
typedef int T;
const T acT = 9; // A constant of type 'T'
const T* pcT = &acT; // A pointer to a constant of type 'T'
void func2 ( const T*& rpcT ) // A reference to a pointer to a constant of type 'T'
{
rpcT = pcT;
}
T* pT; // A pointer to a 'T'
void func ()
{
func2 ( pT ); // Should be an error, but isn't detected
*pT = 7; // Invalidly overwrites the constant 'acT'
}
ISO646.H를 사용할 수 없음
/Ze 옵션을 사용할 경우 텍스트 서식의 다음 연산자를 사용하려면 iso646.h를 포함해야 합니다.
&&(and)
&= (and_eq)
& (bitand)
| (bitor)
~ (compl)
! (not)
!= (not_eq)
|| (or)
|= (or_eq)
^ (xor)
^= (xor_eq)
문자열 리터럴 주소의 형식이 const char (*) []가 아닌 const char []임
다음 예제 코드는 /Za가 지정된 경우에는 char const (*)[4]를 출력하지만 /Ze가 지정된 경우에는 char const [4]를 출력합니다.
#include <stdio.h>
#include <typeinfo>
int main()
{
printf_s("%s\n", typeid(&"abc").name());
}