指针和数组名,还有函数名也遵循隐式和显式类型转换,结构体和联合体不能转换,虽然它们的指导可以转换为其他的指针类型。
一个数组或函数操作指示符为具有类型的任何表达式,在大多数情况下,编译器隐式地转换一个数组的类型,及数组的名子转为数组首元素的指针,数组表达式在以下情景下不能转换为指针:
l 当使用sizeof操作符操作数组时;
l 当使用&操作符时;
l 当使用字符串变量初始化char或wchar_t数组时;
下面的例子示范了隐式转换数组操作符为指针,使用%p打印指针值:
#include <stdio.h>
int *iPtr = 0; // A pointer to int, initialized with 0.
int iArray[ ] = { 0, 10, 20 }; // An array of int, initialized.
int array_length = sizeof(iArray) / sizeof(int); // The number of elements:
// in this case, 3.
printf("The array starts at the address %p./n", iArray);
*iArray = 5; // Equivalent to iArray[0] = 5;
iPtr = iArray + array_length - 1; // Point to the last element of iArray:
// Equivalent to
// iPtr = &iArray[array_length-1];
printf("The last element of the array is %d./n", *iPtr);
在初始化array_length时,表达式sizeof(iArray)取得了数组的长度,而不是指针的大小,然而,在其他三个语句中,指示符iArray隐式地转换为一个指针,如下:
l 在第一次调用printf()时;
l 在使用操作符*时;
l 给iPtr赋值时;
字符数组的名子在字符串操作中也当做指针使用,如下例:
#include <stdio.h>
#include <string.h> // Declares size_t strlen( const char *s )
char msg[80] = "I'm a string literal."; // Initialize an array of char.
printf("The string is %d characters long./n", strlen(msg));
// Answer: 21.
printf("The array named msg is %d bytes long./n", sizeof(msg));
// Answer: 80.
本例中strlen(msg)调用中,标识符msg隐式地转换为指向数组首元素的指针,类型为函数的入参类型const char *, strlen()仅仅统计字符开始到第一个null字符出现时所有的字符个数。
类似地,任何表示函数的表达式,如函数名,也可以隐式地转换为指向函数的指针,同样,当使用地址操作符&时,将不会使用这种转换,sizeof操作符不能使用在函数类型的操作中。下面的例子描述了函数名隐式转换为指针,程序初始化了指向函数的指针,函数的调用在一个循环中。
#include <stdio.h>
void func0( ) { puts("This is the function func0( ). "); } // Two functions.
void func1( ) { puts("This is the function func1( ). "); }
/* ... */
void (*funcTable[2])(void) = { func0, func1 }; // Array of two pointers to
// functions returning void.
for ( int i = 0; i < 2; ++i ) // Use the loop counter as the array index.
funcTable[i]( );