Linux - 面向连接的socket通信流程
字符串的IP与32的IP的转换
说明
网络上的IP都是数字加点(192.168.0.1)构成
struct in_addr结构使用32位的IP,如
IP(C0A80001)是192.168.0.1
函数原型
int inet_aton(const char *cp, struct in_addr *inp)
将a.b.c.d的IP转换为32位的IP,存储在 inp指针里面
char *inet_ntoa(struct in_addr in)
将32位IP转换为a.b.c.d的格式
说明
a代表 ascii,n代表network
面向连接的socket通信流程
服务器程序作用
初始化服务器程序
持续监听一个固定的端口
收到客户的连接后建立一个socket连接
与客户进行通信和信息处理
接收客户通过socket连接发送来的数据,创建一个新的socket,
通过socket连接与客户通信,进行相应处理,并返回处理结果
通信结束后中断与客户的连接(关闭socket)
客户程序作用
初始化客户程序
连接到某个服务器上,建立socket连接
与服务器进行通信和信息处理
接收服务器通过socket连接发送来的数据,进行相应处理
通过socket连接向服务器发送请求信息
通信结束后中断与客户的连接
服务器请求处理流程—循环服务器方案
UDP实现框架
没有一个客户端可以一直占用服务端
只要处理过程不是死循环,则服务器对于每一个客户机的请求总是能够满足
socket(...);
bind(...);
while(1)
{
recvfrom(...);
process(...);
sendto(...);
}
TCP实现框架
每次接受一个客户端连接
完成某客户所有请求后,断开连接
socket(...);
bind(...);
listen(...);
while(1){
accept(...);
{
read(...);
process(...);
write(...);
}
close(...);
}
TCP实现框架
s = socket(...);
bind(s, ...);
listen(s, ...);
while(1) {
ns = accept(s, ...);
if(fork(..)==0) {
read(ns, ...);
process(...);
write(ns, ...);
close(ns);
}
}
}
多线程客户/服务器示例—线程实现
功能描述
客户端
使用线程向服务器发送从标准输入得到的字符
在主线程中将从服务器端返回的字符显示到标准输出
服务器端
将客户端发来的数据原样返回给客户端,每一个客户在服务器上对应一个线程
服务器并发处理优化途径
多路复用I/O
工作机理
进程可在多个socket上等待网络事件,当某个socket发生网络事件时,用户可通过查看网络事件对该socket事件进行I/O操作
要求应用程序不断向内核写入socket描述符,内核还应不断扫描socket集来检测状态
实现途径
预先构造一张描述符表,随后通过select()选择一个已准备好的描述符并返回
应用场景
一个进程中有多个客户连接(存在多个TCP 套接字描述符),select()函数阻塞直到任何一个描述符被激活
避免进程为等待一个已连接上的数据而无法处理其他连接
select()函数
int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout)
nfds:所有被监控文件描述符中最大的一个加1
readfds:所有要读的文件文件描述符的集合
writefds:所有要写的文件文件描述符的集合
exceptfds:其他需要向我们通知的文件描述符
timeout:超时设置
运行机制
select调用进程一直阻塞直到以下情况之一发生
有文件可以读
有文件可以写
超时所设置的时间到
返回结果
-1:表明函数调用发生错误,错误号存放在errno变量中
0:表明发生超时而没有发生其他的错误
大于0:表明事件发生的文件描述符的数目
多路复用I/O相关宏
void FD_SET(int fd, fd_set *fdset)
将fd加入到fdset
void FD_CLR(int fd, fd_set *fdset)
将fd从fdset里面清除
void FD_ZERO(fd_set *fdset)
从fdset中清除所有的文件描述符
int FD_ISSET(int fd, fd_set *fdset)
判断fd是否在fdset集合中
功能
更多的灵活性,可编写TCP/UDP套接字不能实现的功能
可由用户自己创建各个头部
只能由有root权限的用户创建
函数原型
int sockfd(AF_INET, SOCK_RAW, protocol)
根据protocol(如IPPROTO_ICMP, IPPROTO_TCP, IPPROTO_UDP)创建不同类型原始套接字
郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。