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

网络超时检测-setsockopt()

2016-09-27 20:24 429 查看
setsockopt()函数

1、使用setsockopt()实现超时检测时相比其他两种方式的特点是:

只要调用setsockopt函数一次,函数下发所有阻塞函数均可使用,且永久有效。

阻塞函数在到达设置的时间时,会被系统认定为错误,使阻塞函数返回值小于0;

4000

#include <sys/socket.h>

int setsockopt(int socket, int level, int option_name,
const void *option_value, socklen_t option_len);
功能:设置套接字相关的选项信息
参数:
socket: 文件描述符

level: 对应协议层
SOL_SOCKET 应用层
IPPROTO_TCP TCP层
IPPROTO_IP IP层

option_name:选项的名称
SO_BROADCAST 允许发送广播 int
SO_REUSEADDR 允许重复使用地址 int
SO_SNDBUF 获取发送缓冲器大小
SO_RCVBUF 获取接收缓冲区大小
SO_RCVTIMEO 设置接收超时时间
SO_SNDTIMEO 设置发送超时时间

option_value:对应选项的值

option_len:大小

返回值:
成功:0
失败:-1


SO_RCVTIMEO 设置接收超时时间

+++++++++++++++++++++++++++++++++++++

struct timeval {

int tv_sec; 秒

int tv_usec; 微秒

};

+++++++++++++++++++++++++++++++++++++

//使用setsockopt实现网络超时检测

//setsockopt函数只要调用一次,下方所有的阻塞函数都会使用,并永久有效

struct timeval out_time;

out_time.tv_sec = 5;

out_time.tv_usec = 0;

if(setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, &out_time, sizeof(out_time)) < 0)

{

errlog(“fail to setsockopt”);

}

下面上实例:

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

#define N 128

#define errlog(errmsg) do{perror(errmsg); exit(1);}while(0)

int main(int argc, const char *argv[])
{
int sockfd;
struct sockaddr_in serveraddr, clientaddr;
int acceptfd;
socklen_t addrlen = sizeof(struct sockaddr_in);

char buf
= {};

//创建套接字
if((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
{
errlog("fail to socket");
}

//填充网络信息结构体
//inet_addr 将点分十进制转化成网络字节
//htons表示将主机字节序转化成网络字节序
//atoi 将字符串转化成整型数据
serveraddr.sin_family = AF_INET;
serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
serveraddr.sin_port = htons(atoi(argv[2]));

//将套接字与IP地址和端口号绑定
if(bind(sockfd, (struct sockaddr *)&serveraddr, sizeof(serveraddr)) < 0)
{
errlog("fail to bind");
}

//将套接字设置为被动监听状态
if(listen(sockfd, 10) < 0)
{
errlog("fail to listen");
}

//使用setsockopt实现网络超时检测
//setsockopt函数只要调用一次,下方所有的阻塞函数都会使用,并永久有效

struct timeval out_time;     // 结构体原型可以用“ vim -t timeval ”来查看
out_time.tv_sec = 5;          //设置超时检测的时间为5秒
out_time.tv_usec = 0;

if(setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, &out_time, sizeof(out_time)) < 0)
{
errlog("fail to setsockopt");
}
//setsockopt函数在调用之后,所有的阻塞等待函数就上了一个计时器,例如设定时间为5秒,在阻塞了4秒后,第5秒
//仍没有接收到数据的话,系统会认定这个阻塞函数错误。此时,函数(以accept函数为例,其他函数暂未实验)返回
//失败 -1 ,在返回失败后,进入if(errno == 11)函数内,打印timeout.
//验证:把if(errno == 11){printf("timeout\n");}删除后,调用函数,程序在阻塞4秒后,在第5秒时,
//执行了errlog("fail to accept"),在终端打印了错误信息后,退出了程序。
while(1)
{
//接收客户端的连接请求
if((acceptfd = accept(sockfd, (struct sockaddr *)&clientaddr, &addrlen)) < 0)
{
//printf("errno = %d\n", errno); errno =11
//代表此时超时检测错误信息,不应退出
if(errno == 11)
{
printf("accept timeout ...\n");
}
else
{
errlog("fail to accept");
}
}
else
{

printf("%s ---> %d\n", inet_ntoa(clientaddr.sin_addr), ntohs(clientaddr.sin_port));

//与客户端进行通信
while(1)
{
if(recv(acceptfd, buf, N, 0) < 0)
{
//printf("errno = %d\n", errno);
if(errno == 11)
{
printf("recv timeout ...\n");
}
else
{
errlog("fail to recv");
}
}
else
{

if(strncmp(buf, "quit", 4) == 0)
{
printf("%s is quited...\n", inet_ntoa(clientaddr.sin_addr));
break;
}
else
{
printf("from client >>> %s\n", buf);

strcat(buf, " from server...");

if(send(acceptfd, buf, N, 0) < 0)
{
errlog("fail to send");
}
}
}
}
}
}

close(sockfd);
close(acceptfd);
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: