新手编程出现关于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);
}
#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);
}
然后如果我再让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指针指向无法存取的内存空间