您的位置:首页 > 理论基础 > 计算机网络

tcp/IP点对点通信程序

2016-05-29 16:14 330 查看
点对点的通信

服务器端与客户端在建立连接之后创建一个进程

服务器端:

子进程用于接收主机的输入并将数据发送出去。父进程用于接收客户端的数据并输出到主机。

  子进程一直等待主机的输入,输入的数据放在发送缓存区。当有输入时,将输入数据发送到客户端,然后清空发送缓冲区。因为如果下一次的输入长度比本次短,那么由于缓冲区中本次接收到的数的后面的数违背清除掉,这些数据将和本次数据一起发送出去。当服务器关闭时,子进程将退出,销毁。

  父进程一直等待接受客户端的数据,当接收到的数据个数为0,即客户端关闭时,父进程将退出while循环,并销毁进程。当接受到的数据大于0时,则将接受大的数据输出到主机。

#include<stdio.h>
#include<stdlib.h>
#include<errno.h>
#include<string.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<sys/un.h>
#include<sys/wait.h>      //*进程用的头文件*/
#include<netinet/in.h>
#include<arpa/inet.h>

#define MAXLINE  50  //通信内容的最大长度
int main()
{
int sock_fd,new_fd;//sock_fd用于监听,new_fd用于连接
struct sockaddr_in srv_addr;//服务器的地址信息
struct sockaddr_in client_addr;//客户机的地址信息
int size; //地址结构数据的长度
pid_t  pid;  //子进程id
ssize_t n;
char buf[MAXLINE]; //用于存放通信的内容

/*创建套接字*/
sock_fd=socket(AF_INET,SOCK_STREAM,0);//采用IPv4协议
if(sock_fd==-1)
{
perror("creat socket failed");
exit(1);
}

/*服务器地址参数*/
srv_addr.sin_family=AF_INET;
srv_addr.sin_port=htons(3490);
srv_addr.sin_addr.s_addr=htonl(INADDR_ANY);
bzero(&srv_addr.sin_zero,sizeof(struct sockaddr_in));//bzero位清零函数,将sin_zero清零,sin_zero为填充字段,必须全部为零

int on=1; //表示开启reuseaddr
if(setsockopt(sock_fd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on))<0)  //打开地址、端口重用
perror("setsockopt");

/*绑定地址和端口*/
if(bind(sock_fd,(struct sockaddr*)&srv_addr,sizeof(struct sockaddr))==-1)
{
perror("bind failed");
exit(1);
}

/*连接到服务器
if(connect(sock_fd,(struct sockaddr*)&srv_addr,sizeof(sock_fd))==-1)
{
perror("bind failed");
exit(1);
}*/

/*设置监听模式,等待客户机的监听*/
if((listen(sock_fd,5))==-1)
{
perror("listen failed");
exit(1);
}

/*接受连接,采用非阻塞是的模式调用accep*/
//while(1)
//{
size=sizeof(struct sockaddr_in);
new_fd=accept(sock_fd,(struct sockaddr*)&client_addr,&size);
if(new_fd==-1)
{
perror("accept failed");
//continue;//restart accept when EINTR
}

printf("server:got connection from IP= %s prot= %d \n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));//连接成功,打印客户机IP地址和端口号
/*char *inet_nota(struct sockaddr_in in);
头文件:
arpa/inet.h
Winsock2.h
参数:
一个网络上的IP地址
返回值:
如果正确,返回一个字符指针,指向一块存储着点分格式IP地址的静态缓冲区(同一线程内共享此内存);错误,返回NULL。
uint31_t ntohs(uint32_t net32bitvalue);
头文件:
#include<netinet/in.h>
把net32bitvalue有网络字节序转换为主机字节序。
*/
if(send(new_fd,"Hello client,I am 192.168.229.125!\n",50,0)==-1)  //192.168.229.125为子进程IP,可更改
perror("send failed");

pid=fork();  //父进程建立套接字的连接之后,创建子进程用于通信
if(pid<0)
perror("fork error\n");
if(!pid)//创建新的子进程,用于发送数据
{
// close(sock_fd);//子进程不需要监听,所以子进程关闭监听套接字
while(fgets(buf,sizeof(buf),stdin)!=NULL)
{
write(new_fd,buf,strlen(buf));
memset(buf,0,sizeof(buf));   //清空,以免和下一次混淆
// exit(EXIT_SUCCESS);
}
exit(EXIT_SUCCESS);
}
else   //f=父进程用于接收数据
{
char recvbuf[50];
//close(new_fd);//父进程不需要连接,所以关闭连接套接字
while(1)
{
memset(recvbuf,0,sizeof(recvbuf));
n=read(new_fd,recvbuf,MAXLINE);
if(n==0)
{
printf("peer closed\n");
break;
}
else if(n<0)
perror("read from client error");
fputs(recvbuf,stdout);
}
exit(EXIT_SUCCESS);

}

//while(waitpid(-1,NULL,WNOHANG)>0);//等待子进程结束,进行新的连接
//}
return 0;
}


客户端:

子进程用于接收服务器的数据并输出到主机。父进程用于接收主机输入的数据并发送到服务器。

#include<stdio.h>
#include<stdlib.h>
#include<errno.h>
#include<string.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<sys/un.h>
#include<sys/wait.h>      //*进程用的头文件*/
#include<netinet/in.h>
#include<arpa/inet.h>

#define MAXBYTEMUN   50
int main(int argc,char *argv[])
{
int sock_fd,numbytes;
char buf[MAXBYTEMUN];
struct hostent *he;
struct sockaddr_in client_addr;//客户机的地址信息
ssize_t n;

if(argc!=2)
{
fprintf(stderr,"usage: client IPAddress\n");   //执行客户端程序时,输入客户端程序名称和其IP地址
exit(1);
}

/*创建套接字*/
sock_fd=socket(AF_INET,SOCK_STREAM,0);//采用IPv4协议
if(sock_fd==-1)
{
perror("creat socket failed");
exit(1);
}

/*服务器地址参数*/
client_addr.sin_family=AF_INET;
client_addr.sin_port=htons(3490);
client_addr.sin_addr.s_addr=inet_addr(argv[1]);
bzero(&client_addr.sin_zero,sizeof(struct sockaddr_in));//bzero位清零函数,将sin_zero清零,sin_zero为填充字段,必须全部为零

/*连接到服务器*/
if(connect(sock_fd,(struct sockaddr*)&client_addr,sizeof(struct sockaddr))==-1)
{
perror("connect failed");
exit(1);
}
if((numbytes=recv(sock_fd,buf,MAXBYTEMUN,0))==-1)
{
perror("receive failed");
exit(1);
}
buf[numbytes]='\0';//在字符串末尾加上\0,否则字符串无法输出
printf("Received: %s\n",buf);

pid_t pid;
pid=fork();
if(!pid)//创建新的子进程,用于接收数据
{
char recvbuf[50];
while(1)
{
memset(recvbuf,0,sizeof(recvbuf));
n=read(sock_fd,recvbuf,MAXBYTEMUN);
if(n==0)
{
printf("peer closed\n");
break;
}
else if(n<0)
perror("read from server error");
fputs(recvbuf,stdout);
}
//exit(EXIT_SUCCESS);
close(sock_fd);
}
else   //f=父进程用于发送数据
{
//close(sock_fd);//父进程不需要连接,所以关闭连接套接字
while(fgets(buf,sizeof(buf),stdin)!=NULL)
{
write(sock_fd,buf,strlen(buf));
memset(buf,0,sizeof(buf));   //清空,以免和下一次混淆
// exit(EXIT_SUCCESS);
}
//exit(EXIT_SUCCESS);
close(sock_fd);
}
/*接受数据
if((numbytes=recv(sock_fd,buf,MAXBTYEMUN,0))==-1)
{
perror("receive failed");
exit(1);
}

buf[numbytes]='\0';//在字符串末尾加上\0,否则字符串无法输出
printf("Received: %s\n",buf);
close(sock_fd);*/
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: