Microsoft 對 C 和 C++ 的擴充功能
Visual C++ 如下擴充 ANSI C 或 ANSI C++ 標準。
Keywords
數個關鍵字加入。 在 C + + 關鍵字中的清單中,有的關鍵字兩個前置底線是 Visual C++ 擴充功能。
static const 整數 (或列舉) 成員的類別外定義
根據這個標準 (/Za),您必須將資料成員的類別外定義,如下所示:
class CMyClass {
static const int max = 5;
int m_array[max];
}
...
const int CMyClass::max; // out of class definition
在 /Ze 之下,類別外定義對於靜態資料成員、const 整數資料成員和 const 列舉資料成員是選擇性的。 靜態的整數和只列舉和常數可以在類別的初始設定式;初始化運算式必須是常數運算式。
若要避免此錯誤,當類別外定義在標頭檔中提供,且標頭檔在多個原始程式檔中,請使用 >selectany。 例如:
__declspec(selectany) const int CMyClass::max = 5;
轉換
編譯器支援這類非 ANSI 轉換:
產生左值的非 ANSI 轉換。 例如:
char *p; (( int * ) p )++;
注意事項 此擴充功能可在只 C 語言。您可以使用下列 ANSI C 標準格式在 C++ 程式碼修改指標,就好像指標不同型別。
以上範例可以重寫如下符合 ANSI C 標準。
p = ( char * )(( int * )p + 1 );
一個函式指標的非 ANSI 轉換到資料指標的。 例如:
int ( * pfunc ) (); int *pdata; pdata = ( int * ) pfunc;
在您將它轉型為資料指標之前,要執行相同的轉換同時維持 ANSI 相容性,可以轉型成函式指標設定為 uintptr_t :
pdata = ( int * ) (uintptr_t) pfunc;
可變長度引數清單
編譯器支援指定引數的變數數字的函式宣告子,後面接著提供型別的函式定義:
void myfunc( int x, ... );
void myfunc( int x, char * c )
{ }
單行註解
C 編譯器支援單行註解,使用兩個正斜線 (/) 字元,引入:
// This is a single-line comment.
範圍
C 編譯器支援下列範圍相關功能。
重新定義外部項目為 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; }
可變大小陣列 (Unsized Array) 做為結構和等位的最後一個欄位:
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 時,支援內嵌產生 x86 專屬資訊 > atan、atan2、cos、exp、log、log10、sin、sqrt 和 tan 函式 x86 專屬資訊結束。 對於 C,使用這些內建函式時將會失去 ANSI 一致性,因為它們不會設定 errno 變數。
傳遞非常數指標參數給需要參考常數指標參數的函式
這是對 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());
}