网络IPC:套接字
用socket实现两个不同的主机之间的通信(涉及到一些基本的计算机网络知识 略过。。)
服务器端:
1.socket函数:生成一个套接字
int socket(int domain,int type,int protocol);
参数解析:domain:{AF_INET:Ipv4网络协议,AF_INET6:Ipv6网络协议}
type:{tcp:SOCK_STREAM,udp:SOCK_DGRAM}
protocol:指定socket所使用的传输协议编号,,一般为0;
2.bind函数
将套接字与地址关联
int bind(int sockfd,const struct sockaddr *addr,socklen_t len);
sockfd:套接字 *addr:地址结构的地址 len:地址结构的长度
IPV4中,套接字地址用结构sockaddr_in表示:
struct sockaddr_in {
sa_family_t sin_family; //通信域,一般为AF_INET
in_port_t sin_port; //接口地址,二进制
struct in_addr sin_addr; //iP地址,二进制
}
3.listen函数:使服务器ip和这个端口处于监听状态,如果网络中某一客户机有连接请求,则接受请求。
int listen(int sockfd,int backlog);
sockfd:套接字 backlog:服务器能接受的最大请求,一般为10,最大为128
4.accpet函数:接受客户端的请求,建立与客户机端的通信连接。当服务器处于监听状态时,客户端有连接请求,服务器不会马上处理,而是把这一个请求添加到等待队列中去,等到服务器空闲时再处理。处理时会生成一个新的套接字,这个套接字用于服务器和该客户端进行通信。原来的那个socket套接字继续用于监听。
int accept(int s,struct sockaddr *addr,int *addrlen)
s:socket返回值 addr:结构体指针变量,和bind是同种类型(系统会把远程客户机的IP和端口号放到这个指针变量中去) addrlen:结构体长度
成功的话返回的值是新的socket套接字
5.recv函数:用新的套接字来接受远程客户端所传来的数据,并且将数据存储到参数buf中去
原型:int recv(int sockfd,void *buf,int len,unsigned int flags);
参数:sockfdà为前面accept的返回值.即new_fd,也就是新的套接字。
bufà表示缓冲区
lenà表示缓冲区的长度
flagsà通常为0
成功会返回接受数据的长度。
6.send函数:用新的套接字发送数据给远程客户端
原型:int send(int s,const void * msg,int len,unsigned int flags);
参数:s为前面accept的返回值.即new_fd
msg一般为常量字符串
len表示长度
flags通常为0
客户端:
1. connect函数:用来请求连接远程服务器,将参数sockfd 的socket 连至参数serv_addr 指定的服务器IP和端口号上去。
原型:int connect (int sockfd,struct sockaddr * serv_addr,int addrlen);
参数:sockfd为前面socket的返回值,即sfd
serv_addr为结构体指针变量,存储着远程服务器的IP与端口号信息。
addrlen表示结构体变量的长度
返回值:成功则返回0,失败返回-1
2.close函数:当使用完文件后若已不再需要则可使用close()关闭该文件,并且close()会让数据写回磁盘,并释放该文件所占用的资源
原型:int close(int fd);
参数:fd为前面的sfd,new_fd
返回值:若文件顺利关闭则返回0,发生错误时返回-1
sample:一个简单的通信过程。ps:该程序没有用recv函数和send函数,而是用了read和write函数
服务器端:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/types.h>
#include<arpa/inet.h>
#include<sys/socket.h>
#include<netinet/in.h>
int main(int argc,char *argv[])
{
int fd_listen;
fd_listen=socket(AF_INET,SOCK_STREAM,);//套接字 if(fd_listen==-)
{
perror("socket");
exit();
}
struct sockaddr_in seraddr; //地址结构体
memset(&seraddr,,sizeof(seraddr));
seraddr.sin_family=AF_INET;
seraddr.sin_port=htons();//端口
seraddr.sin_addr.s_addr=inet_addr("192.168.1.182");//ip地址
if(-==bind(fd_listen,(const struct sockaddr*)&seraddr,(socklen_t)sizeof(seraddr)))
{
perror("bind");
close(fd_listen);
exit();
}
listen(fd_listen,); //监听函数 struct sockaddr_in peeraddr;
memset(&peeraddr,,sizeof(peeraddr));
socklen_t len=sizeof(peeraddr);
int fd_peer=accept(fd_listen,(struct sockaddr*)&peeraddr,&len); printf("who:%s:%d\n",inet_ntoa(peeraddr.sin_addr),ntohs(peeraddr.sin_port));
if(fd_peer==-)
{
perror("accept");
close(fd_listen);
exit();
}
char buf[]="";
int readn=read(fd_peer,buf,);//从新套接字中读取数据
printf("readn:%d,msg:%s\n",readn,buf);
char *p="-------------------";
write(fd_peer,p,strlen(p));
close(fd_listen);
close(fd_peer);
return ;
}
客户端:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<netinet/in.h>
int main(int argc,char *argv[])
{
int sfd=socket(AF_INET,SOCK_STREAM,);
if(sfd==-)
{
perror("socket");
exit();
}
struct sockaddr_in peeraddr;
peeraddr.sin_family=AF_INET;
peeraddr.sin_port=htons();
peeraddr.sin_addr.s_addr=inet_addr("192.168.1.182"); if(-==connect(sfd,(struct sockaddr*)&peeraddr,sizeof(peeraddr)))//请求连接,把自己的信息添加到套接字中去
{
perror("connect");
close(sfd);
}
char *p="hello world";
write(sfd,p,strlen(p));
char buf[]="";
int readn=read(sfd,buf,);
printf("readn:%d: %s\n",readn,buf);
close(sfd);
return ;
}
总结:本程序模拟TCP协议实现了服务器端和客户机端的简单通信,介绍了整个过程。