编写代码时,有时候需要找到某个结构的定义,例如类型,成员变量名.首先使用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.