我曾在引起分段错误时用谷歌搜索,但我无法纠正我的代码中的错误。我正在使用select函数来找出可读取的文件描述符。 c,当我们具有用于ipv4地址长度的宏时,我无法理解socklen_t结构的目的。
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/socket.h>
#include<sys/types.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<netdb.h>
#define PORT "9034"
int main()
{
fd_set master,read_fds;
int file_descriptor,new_fd,server,yes=1,error,fd_max,i;
char ip[INET_ADDRSTRLEN],data[1024];
struct addrinfo hints,*p;
struct sockaddr_in addr;
socklen_t addrlen; //?
memset(&hints,0,sizeof(struct addrinfo));
FD_ZERO(&master);
FD_ZERO(&read_fds);
hints.ai_family=AF_INET;
hints.ai_socktype=SOCK_STREAM;
hints.ai_protocol=AI_PASSIVE;
file_descriptor=getaddrinfo(NULL,PORT,&hints,&p);
printf("%s",inet_ntop(AF_INET,p->ai_addr,ip,addrlen));
server=socket(p->ai_family,p->ai_socktype,p->ai_protocol);
setsockopt(server,SOL_SOCKET,SO_REUSEADDR,&yes,sizeof(int));
listen(server,5);
FD_SET(server,&master);
fd_max=server;
while (1)
{
read_fds=master;
select(fd_max+1,&read_fds,NULL,NULL,NULL);
for(i=0;i<fd_max;i++)
{
if(i==server)
{
new_fd=accept(server,(struct sockaddr *)&addr,&addrlen);
FD_SET(new_fd,&master);
printf("Connected to the client %s on socket %d",inet_ntop(AF_INET,(struct sockaddr *)&addr,ip,INET_ADDRSTRLEN),new_fd);
if(new_fd>fd_max)
fd_max=new_fd;
}
else
{
recv(i,data,sizeof(data),0);
printf("%s",data);
}
}
}
}
答案 0 :(得分:0)
根据您的评论,似乎您正在使用的某些实用程序可能未正确定义。对于要放入项目的库,请确保使用正确的(和相应的)头文件。即,对于POSIX开发,建议使用以下方法:
/* According to POSIX.1-2001 */
#include <sys/select.h>
在您的select.h
语句中看不到#include
文件。
答案 1 :(得分:0)
崩溃在这里:
file_descriptor=getaddrinfo(NULL,PORT,&hints,&p);
printf("%s",inet_ntop(AF_INET,p->ai_addr,ip,addrlen));
请注意,在第二行中,您正在取消引用指针p
,但是p
为NULL,因为getaddrinfo()
失败并且未将其设置为有效地址,因此请尝试取消引用会调用未定义的行为,在这种情况下会导致崩溃。
再说一次,我可以通过在程序的各个位置插入临时printf()
语句并查看打印了(或未打印)的内容来调试此问题。特别是:
printf("k1\n");
file_descriptor=getaddrinfo(NULL,PORT,&hints,&p);
printf("k2 file_descriptor=%i p=%p\n", file_descriptor, p);
printf("%s",inet_ntop(AF_INET,p->ai_addr,ip,addrlen));
printf("k3\n");
...当我运行类似的代码时,我看到了以下输出:
k1
k2 file_descriptor=12 p=0x0
Segmentation fault: 11
...这使问题显而易见-p在“ k2”行中为NULL(aka 0x0),并且崩溃发生在“ k3”行被打印之前,因此崩溃肯定在您的{{1 }}调用,并且由于NULL指针取消引用。
要解决崩溃问题,您需要检查printf()
的返回值以确保其返回零/成功,并在返回非零/错误时优雅地处理它,而不仅仅是假设将永远成功。 (通常,您需要对返回成功/失败值的所有函数调用执行此操作)