首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > C语言 >

新手编程出现关于fork()的小疑点

2013-12-30 
新手编程出现关于fork()的小问题本帖最后由 u013301068 于 2013-12-27 22:41:25 编辑代码很简单,就是TCP网

新手编程出现关于fork()的小问题
本帖最后由 u013301068 于 2013-12-27 22:41:25 编辑 代码很简单,就是TCP网络通讯
先上代码好了

Server:

#include<stdio.h> 
#include<stdlib.h> 
#include<string.h> 
#include<arpa/inet.h> 
#include<netdb.h> 
#include<netinet/in.h> 
#include<sys/types.h> 
#include<sys/socket.h> 
#include<unistd.h> 

//Vous pouvez changer les paramètres suivantes
#define MAX_MSG_SIZE 10000 
#define SERVER_PORT 50001 
#define CLIENT_PORT 50002 
#define SERVER_ADDR 127.0.0.1 
//Le nombre Maximum de connexions arrivées
#define BACKLOG 5 

int main(void) 

int ser_sockfd,cli_sockfd; 
struct sockaddr_in ser_addr; 
struct sockaddr_in cli_addr; 
char msg[MAX_MSG_SIZE]; 
pid_t fpid;

//Créer le socket d'écoute
ser_sockfd=socket(AF_INET,SOCK_STREAM, 0); 
if(ser_sockfd<0) 

perror("Socket"); 
exit(1); 


int addrlen=sizeof(struct sockaddr_in); 
memset(&ser_addr, 0, addrlen); 
ser_addr.sin_family=AF_INET; 
ser_addr.sin_addr.s_addr=htonl(INADDR_ANY); 
ser_addr.sin_port=htons(SERVER_PORT); 
if(bind(ser_sockfd, (struct sockaddr*)&ser_addr, sizeof(struct sockaddr_in))<0) 

perror("Bind"); 
exit(1); 


if(listen(ser_sockfd,BACKLOG)<0) 

perror("Listen"); 
close(ser_sockfd); 
exit(1); 


while(1) 

cli_sockfd=accept(ser_sockfd, (struct sockaddr*)&cli_addr, &addrlen); 
if(cli_sockfd<=0) 
perror("Accept"); 
else 

fpid=fork();
if(fpid<0)
{
printf("Erreur de processus fils");
exit(1);
}
else if(fpid==0)
{
recv(cli_sockfd,msg,MAX_MSG_SIZE,0); 
printf("%d, Recevoir un message de: %s\n", fpid, inet_ntoa(cli_addr.sin_addr)); 
                    printf("%d, %s\n", fpid, msg); 
strcpy(msg,"Bonjour! Je suis server!"); 
                    send(cli_sockfd, msg, sizeof(msg), 0); 
                    close(cli_sockfd); 
}


close(ser_sockfd); 
}


Client:

#include<stdio.h> 
#include<stdlib.h> 
#include<string.h> 
#include<arpa/inet.h> 
#include<netdb.h> 
#include<netinet/in.h> 
#include<sys/types.h> 
#include<sys/socket.h> 
#include<unistd.h> 

#define MAX_MSG_SIZE 10000 
#define SERVER_PORT 50001 
#define CLIENT_PORT 50002 
#define SERVER_ADDR 127.0.0.1 

//int main(void)
int main(int argc, char *argv[])

int cli_sockfd; 
int addrlen; 
char seraddr[14]; 
struct hostent *hostp; 
struct sockaddr_in ser_addr, cli_addr; 
char msg[MAX_MSG_SIZE]; 

if(argc<2){ 
printf("usage: %s hostname port \n", argv[0]); 
exit(1); 


cli_sockfd=socket(AF_INET, SOCK_STREAM, 0); 
if(cli_sockfd<0){ 
perror("Socket"); 
exit(1); 


addrlen=sizeof(struct sockaddr_in);
memset(&ser_addr, 0, addrlen); 
cli_addr.sin_family=AF_INET; 
cli_addr.sin_addr.s_addr=htonl(INADDR_ANY); 
cli_addr.sin_port=htons(CLIENT_PORT); 
if(bind(cli_sockfd, (struct sockaddr*)&cli_addr, addrlen)<0){ 
perror("Bind"); 
exit(1); 


memset(&ser_addr, 0, addrlen); 
ser_addr.sin_family=AF_INET; 
hostp=(struct hostent *)gethostbyname(argv[1]); 
//ser_addr.sin_addr.s_addr=inet_addr("127.0.0.1");
ser_addr.sin_addr=*(struct in_addr *) (hostp->h_addr_list[0]); 
//ser_addr.sin_port=htons(SERVER_PORT);
ser_addr.sin_port=htons(atoi(argv[2]));
if(connect(cli_sockfd, (struct sockaddr*)&ser_addr, addrlen)<0){ 
perror("Connection"); 
close(cli_sockfd); 
exit(1); 


strcpy(msg,"Bonjour! Je suis client"); 
send(cli_sockfd,msg,sizeof(msg),0); 
recv(cli_sockfd,msg,MAX_MSG_SIZE,0); 
printf("%s\n",msg); 
close(cli_sockfd); 
}


之后的输出结果:

0, Recevoir un message de: 127.0.0.1
0, Bonjour! Je suis client



然后如果我再让Client运行一次就会出现错误:

Bind: Address already in use

这是第一个问题;

另外,如果我把Server中的if(fpid==0)删掉,显示结果为:

4539, Recevoir un message de: 127.0.0.1
4539, Bonjour! Je suis client
0,Recevoir un message de: 127.0.0.1
0, 

对于最后一行不是很理解,如果说fork是复制下面的语句,然后在子进程中运行,那为啥不会再出现一次 Bonjour! Je suis client呢?

求教ing。。。
[解决办法]
1 因为这个时候套接字进入的是半关闭状态,也就是主动关闭的一方进去了TIME_WAIT状态,
  中间有2MSL时间的,可以通过setsockopt函数设置下REUSEADDR 地址服用!
2 没看到你S端怎么发送消息的!

[解决办法]
int setsockopt(int s,int level,int optname,const void * optval,,socklen_toptlen);
函数说明 
setsockopt()用来设置参数s所指定的socket状态。参数level代表欲设置的网络层,一般设成SOL_SOCKET以存取socket层。参数optname代表欲设置的选项,有下列几种数值:
SO_DEBUG 打开或关闭排错模式
SO_REUSEADDR 允许在bind()过程中本地地址可重复使用
SO_TYPE 返回socket形态。
SO_ERROR 返回socket已发生的错误原因
SO_DONTROUTE 送出的数据包不要利用路由设备来传输。
SO_BROADCAST 使用广播方式传送
SO_SNDBUF 设置送出的暂存区大小
SO_RCVBUF 设置接收的暂存区大小
SO_KEEPALIVE 定期确定连线是否已终止。
SO_OOBINLINE 当接收到OOB 数据时会马上送至标准输入设备
SO_LINGER 确保数据安全且可靠的传送出去。
参数 
optval代表欲设置的值,参数optlen则为optval的长度。
返回值 
成功则返回0,若有错误则返回-1,错误原因存于errno。
附加说明 
EBADF 参数s并非合法的socket处理代码
ENOTSOCK 参数s为一文件描述词,非socket
ENOPROTOOPT 参数optname指定的选项不正确。
EFAULT 参数optval指针指向无法存取的内存空间

热点排行