编写代码时,有时候需要找到某个结构的定义,例如类型,成员变量名.首先使用gcc -E file.cpp -o file.ii进行预编译,然后使用more命令打开file.ii,可以看到所有预编译的头文件所在的全路径.

打开相应的头文件,找到相应的结构成员.不过标准库中的有些结构使用宏,不是那么直观,这很讨厌,下面以bind函数使用的socketaddr这个结构为例.下面是代码(部分):
#include <stdlib.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <netinet/in.h>
#include <unistd.h>
int main()
{
        printf("this is test server\r\n");
        int serversockd = socket(AF_UNIX,SOCK_STREAM,0);
 
        struct sockaddr serveraddr;
        serveraddr.sa_family = AF_UNIX;
        strcpy(serveraddr.sa_data,"10.1.1.26");
        int ret = bind(serversockd, &serveraddr, sizeof(serveraddr));
        if (0 != ret )
                printf("Error \r\n");
        ...
}
使用gcc -E file.cpp -o file.ii命令,然后打开file.ii,应该能够看到很多头文件的全路径路径,下面是几个例子:
...
# 1 "/usr/include/stdlib.h" 1 3 4
# 25 "/usr/include/stdlib.h" 3 4
# 1 "/usr/include/features.h" 1 3 4
# 323 "/usr/include/features.h" 3 4
# 1 "/usr/include/bits/predefs.h" 1 3 4
# 324 "/usr/include/features.h" 2 3 4
# 356 "/usr/include/features.h" 3 4
...
可以看到这段代码引用的头文件都在/usr/include及其子目录中,如果不做说明,只说打开文件就指在/usr/include/或其子目录中打开.
 
先看一下socketaddr的定义,<sys/socket.h>中有这么一句,#include <bits/socket.h>,打开bits/socket.h,这个文件中有这么一段:
/* Structure describing a generic socket address.  */
struct sockaddr
  {
    __SOCKADDR_COMMON (sa_);    /* Common data: address family and length.  */
    char sa_data[14];           /* Address data.  */
  };
__SOCKADDR_COMMON是一个宏,这个宏的定义在<bits/sockaddr.h>中,打开该文件,可以看到这么一句:
#define __SOCKADDR_COMMON(sa_prefix) \
  sa_family_t sa_prefix##family
宏替换__SOCKADDR_COMMON (sa_);后的结果就是:sa_family_t sa_family,也就是说struct sockaddr的定义其实是这样的:
struct sockaddr
  {
    sa_family_t sa_family    /* Common data: address family and length.  */
    char sa_data[14];           /* Address data.  */
  };
 
还可以在<sys/un.h>看到sockaddr_un的定义,下面是它的结构:
struct sockaddr_un
  {
    __SOCKADDR_COMMON (sun_);
    char sun_path[108];         /* Path name.  */
  };
它也使用了__SOCKADDR_COMMON.
 
可以在<netinet/in.h>看到sockaddr_in的定义,下面是它的结构:
struct sockaddr_in
  {
    __SOCKADDR_COMMON (sin_);
    in_port_t sin_port;                 /* Port number.  */
    struct in_addr sin_addr;            /* Internet address.  */
 
    /* Pad to size of `struct sockaddr'.  */
    unsigned char sin_zero[sizeof (struct sockaddr) -
                           __SOCKADDR_COMMON_SIZE -
                           sizeof (in_port_t) -
                           sizeof (struct in_addr)];
  };
它也使用了__SOCKADDR_COMMON.