linux中的分段错误(C中的套接字编程(TCP))
作者:互联网
我只是通过一些网站学习Linux上的套接字编程,这是使用TCP在服务器端的部分代码:
#define BufferLength 100
#define SERVPORT 3111
int main()
{
/* Variable and structure definitions. */
int sd, sd2, rc, length = sizeof(int);
int totalcnt = 0, on = 1;
char temp;
char buffer[BufferLength];
struct sockaddr_in serveraddr;
struct sockaddr_in their_addr;
fd_set read_fd;
/* Get a socket descriptor */
if((sd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
{
perror("Server-socket() error");
exit (-1);
}
else
printf("Server-socket() is OK\n");
/* Allow socket descriptor to be reusable */
if((rc = setsockopt(sd, SOL_SOCKET, SO_REUSEADDR, (char *)&on, sizeof(on))) < 0)
{
perror("Server-setsockopt() error");
close(sd);
exit (-1);
}
else
printf("Server-setsockopt() is OK\n");
/* bind to an address */
memset(&serveraddr, 0x00, sizeof(struct sockaddr_in));
serveraddr.sin_family = AF_INET;
serveraddr.sin_port = htons(SERVPORT);
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);
printf("Using %s, listening at %d\n", inet_ntoa(serveraddr.sin_addr), SERVPORT);
/* continue */
}
当我执行最后一行(printf(“ using ……”))时,出现了分段错误,为什么?谢谢.
解决方法:
所示代码缺少#include任何标头,因此由于某些未定义的符号,它无法编译.
但是,如果您错过了代码所引用的任何库函数的原型,它将进行编译,这将导致假定任何函数都返回int.
后一个事实可能是致命的,也可能不是致命的.
至少在64位系统上,将inet_ntoa()用作printf()的参数是致命的,因为在64位系统上,最有可能返回64位(字符指针)值(但32位int ).因此(假设原型未命中)在生成代码时,编译器假定inet_ntoa()返回32位int,这将导致“斩除”返回地址的最高32位.尝试从此类“已损坏”的(因此很可能是)无效地址进行printf()会引发未定义的行为,并且在您的情况下会导致观察到分段冲突.
要解决此问题,请通过添加以下内容添加相关的原型(用于inet_ntoa()):
#include <arpa/inet.h>
编译器应该已经对此发出警告.要为gcc启用所有编译器警告,请使用-Wall -Wextra -pedantic选项.认真对待此类警告.
标签:sockets,tcp,c-3,linux,segmentation-fault 来源: https://codeday.me/bug/20191120/2043200.html