网络编程(五)

网络编程(五)

  • 网络服务器超时检测
    • 使用select进行超时检测
    • 套接字属性
      • **getsockopt:获取socket软通道的某项属性值**
      • setsockopt:设置socket软通道的某项属性值**(socket建立之后就可使用)
    • 信号
      • **signal():信号处理函数===set**
      • **sigaction():信号处理函数===set/get**

网络服务器超时检测

在网络通信中,很多操作会使得进程阻塞
TCP套接字中的recv/accept/connect
UDP套接字中的recvfrom
超时检测的必要性:
1、避免进程在没有数据时无限制地阻塞
2、当设定的时间到时,进程从原操作返回继续运行

使用select进行超时检测

如果该服务器使用select实现的,可以使用select的最后一个参数做超时检测

#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
int select(int nfds, fd_set *readfds, fd_set *writefds,fd_set *exceptfds, struct timeval *timeout);
//参数1:需要监控的最大文件描述符的个数,总数
//参数2:指针指向读文件描述符集合表
//参数3:指针指向写文件描述符集合表
//参数4:指针指向异常文件描述符集合表
//参数5:① ② ③
//①NULL—>将select变成阻塞函数
//②0—>仅仅检测文件描述符的状态,然后立即返回,不会置位
//③时间值—>表示select的监控超时时间。时间到达之前select监控到有IO通道活跃,select返回通道路数并置位;超时时间到,select没有监控到任何IO通道有数据,那么select返回0
//返回值:成功返回监控到的有数据发生的通道个数;没有活跃则没有返回,则一直阻塞,失败返回-1

*第四个参数struct timeval timeout
在这里插入图片描述
在这里插入图片描述

套接字属性

socket—>有socket层,IP层,tcp/udp层属性

getsockopt:获取socket软通道的某项属性值

#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
int getsockopt(int sockfd, int level, int optname,void *optval, socklen_t *optlen);
//参数1:文件描述符–>标识一个网络软通道
//参数2:level指定控制套接字的层次(SOL_SOCKET,IPPROTO_TCP,IPPROTO_IP)
//参数3:要设置的属性项,根据参数2决定属性项
//参数4:参数3对应的属性值的指针
//参数5:指向参数4大小的指针

setsockopt:设置socket软通道的某项属性值**(socket建立之后就可使用)

#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
int setsockopt(int sockfd, int level, int optname,const void *optval, socklen_t optlen);
//参数1:文件描述符–>标识一个网络软通道
//参数2:level指定控制套接字的层次(SOL_SOCKET,IPPROTO_TCP,IPPROTO_IP)
//参数3:要设置的属性项,根据参数2决定属性项
//参数4:参数3对应的属性值的指针
//参数5:指向参数4大小
//返回值:成功返回0,失败返回-1,并更新errno

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

//超时检测发送方代码
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <strings.h>
#include <string.h>
int main(int argc, const char *argv[])
{//1,创建套接字int socketFd = socket(AF_INET, SOCK_DGRAM, 0);if(socketFd < 0){perror("socket udp client error");return -1;}printf("socket udp client ok!\n");//2,设置服务器的IP 地址和端口struct sockaddr_in addr;bzero(&addr, sizeof(addr));//赋值addr.sin_family = AF_INET;addr.sin_port = htons(8888);//接收方的端口addr.sin_addr.s_addr = inet_addr("224.0.2.88");//组播地址//允许组播int opt = 1;setsockopt(socketFd, IPPROTO_IP, IP_MULTICAST_IF, &opt, sizeof(opt));//3,通信char buf[256] = {0};while(1){bzero(buf, sizeof(buf));printf("推送一条组播消息:");fgets(buf, sizeof(buf), stdin);//从标准输入流中获取数据if(0 == strncasecmp("quit", buf, 4)){printf("结束!\n");bzero(buf, sizeof(buf));sendto(socketFd, buf, strlen(buf), 0, \(struct sockaddr *)&addr, sizeof(addr));break;}//发送消息sendto(socketFd, buf, strlen(buf), 0, \(struct sockaddr *)&addr, sizeof(addr));printf("send multicast message ok!\n");}//关闭套接字close(socketFd);return 0;
}
//超时检测接收方代码
recvfrom进行超时检测
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <strings.h>
#include <string.h>
#include <netinet/in.h>
#include <errno.h>
int main(int argc, const char *argv[])
{//1,创建套接字int socketFd = socket(AF_INET, SOCK_DGRAM, 0);printf("socket server ok!\n");//2,绑定接收方的IP 地址和端口struct sockaddr_in addr;bzero(&addr,sizeof(addr));addr.sin_family = AF_INET;addr.sin_port = htons(8888);//指定的接收方端口,必须和发送方一样addr.sin_addr.s_addr = htonl(INADDR_ANY);//任意地址bind(socketFd, (struct sockaddr *)&addr, sizeof(addr));printf("bind ok!\n");//3,加入到多播组中struct ip_mreq ipaddr;bzero(&ipaddr, sizeof(ipaddr));ipaddr.imr_multiaddr.s_addr = inet_addr("224.0.2.88");ipaddr.imr_interface.s_addr = htonl(INADDR_ANY);setsockopt(socketFd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &ipaddr,sizeof(ipaddr));char buf[256] = {0};//4,接收多播组中的消息//定义存储对方地址信息的结构变量struct sockaddr_in otherAddr;bzero(&otherAddr, sizeof(otherAddr));int len = sizeof(otherAddr);while(1){bzero(buf, sizeof(buf));//设置超时检测struct timeval timeout;bzero(&timeout, sizeof(timeout));timeout.tv_sec = 3;//设置setsockopt(socketFd, SOL_SOCKET, SO_RCVTIMEO,&timeout,sizeof(timeout));//注意:udp服务器一定要存储发送方的地址信息(没有建立过连接,每一次//的发送,都需要知道对方的地址信息!!!)int ret = recvfrom(socketFd, buf, sizeof(buf), 0, \(struct sockaddr *)&otherAddr, &len);if(ret < 0){//判断错误码是否为EAGAINif(errno == EAGAIN){printf("已超时3秒...");}else{perror("recvfrom error");break;}}else if(0 == ret){printf("组中停止消息的发送~\n");break;}else{printf("接收的组播消息:%s\n",buf);}}//5,离开多播组setsockopt(socketFd, IPPROTO_IP, IP_DROP_MEMBERSHIP, &ipaddr,sizeof(ipaddr));//6,关闭套接字close(socketFd);return 0;
}

信号

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

//信号处理函数:
void handler_func(int signum)
{
;
}
//信号行为结构体
struct sigaction {
void (*sa_handler)(int);//信号处理函数,和signal一样的
void (*sa_sigaction)(int, siginfo_t *, void *);
sigset_t sa_mask;
int sa_flags;//标志,可以设置为取消自重启属性
void (*sa_restorer)(void);
};

更改信号默认操作:

signal():信号处理函数===set

signal函数原型:
#include <signal.h>
void (*signal(int signum, void (*handler)(int)))(int);
优化原型:看的更清楚。
//给函数指针类型取别名
#include <signal.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);

功能:进行信号安装
参数:
参数1:需要安装的信号名称
参数2:对于该信号的处理函数(该函数是用户自定义的,且返回值类型void ,形参是一个int的函数的地址)
PS:参数2的位置可以传入:忽略(SIG_IGN) 执行缺省操作(SIG_DFL),信号处理函数的地址
返回值:成功安装完毕的信号处理函数的地址。

sigaction():信号处理函数===set/get

sigaction(信号,处理动作结构,处理动作结构)
① sigaction(SIGALRM, NULL, &act) //获取SIGALRM信号对应的默认处理信息结构
① sigaction(SIGALRM, &act, NULL)//设置SIGALRM信号对应的新的处理信息结构

在这里插入图片描述
在这里插入图片描述

#include <unistd.h>
unsigned int alarm(unsigned int seconds);
//参数:秒数
//当定时器指定的时间到了时,它就向进程发送SIGALARM信号,信号的默认操作是结束进程.
//注意:每个进程只能有一个闹钟时间。如果在调用alarm时,以前已为该进程设置过闹钟时间,而且它还没有超时,则该闹钟时间的剩余时间值作为本次alarm函数调用的值返回,以前登记的闹钟时间则被新值代换。如果有以前登记的尚未超过的闹钟时间,而新设的闹钟时间值为0,则取消以前的闹钟时间,其剩余时间值仍作为函数的返回值。

在这里插入图片描述
在这里插入图片描述

//alarm定时器接收方超时检测代码
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <strings.h>
#include <string.h>
#include <netinet/in.h>
#include <errno.h>
#include <signal.h>
//信号处理函数
void handler_func(int signum)
{printf("已超时...\n");
}
int main(int argc, const char *argv[])
{//1,创建套接字int socketFd = socket(AF_INET, SOCK_DGRAM, 0);printf("socket server ok!\n");//2,绑定接收方的IP 地址和端口struct sockaddr_in addr;bzero(&addr,sizeof(addr));addr.sin_family = AF_INET;addr.sin_port = htons(8888);//指定的接收方端口,必须和发送方一样addr.sin_addr.s_addr = htonl(INADDR_ANY);//任意地址bind(socketFd, (struct sockaddr *)&addr, sizeof(addr));printf("bind ok!\n");//3,加入到多播组中struct ip_mreq ipaddr;bzero(&ipaddr, sizeof(ipaddr));ipaddr.imr_multiaddr.s_addr = inet_addr("224.0.2.88");ipaddr.imr_interface.s_addr = htonl(INADDR_ANY);setsockopt(socketFd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &ipaddr,sizeof(ipaddr));char buf[256] = {0};//4,接收多播组中的消息//定义存储对方地址信息的结构变量struct sockaddr_in otherAddr;bzero(&otherAddr, sizeof(otherAddr));int len = sizeof(otherAddr);//第一步:先获取SIGALRM信号的旧行为struct sigaction act;bzero(&act, sizeof(act));sigaction(SIGALRM, NULL, &act);//第二步:设置处理函数+取消recvfrom的自重启属性act.sa_handler = &handler_func;//act.sa_flags |= SA_RESTART;//启动函数的自重启属性:eg:recvfrom:自重启属性再次陷入阻塞act.sa_flags &= ~SA_RESTART;//取消recvfrom再次陷入阻塞//第三步:将设置完毕的新行为设置给SIGALRM信号sigaction(SIGALRM, &act, NULL);while(1){bzero(buf, sizeof(buf));//使用sigaction函数捕捉SIGALRM信号,给他设置一个处理函数和//取消recv函数自带的自重启属性//设置超时检测alarm(5);//秒到了,会产生SIGALRM信号//注意:udp服务器一定要存储发送方的地址信息(没有建立过连接,每一次//的发送,都需要知道对方的地址信息!!!)int ret = recvfrom(socketFd, buf, sizeof(buf), 0, \(struct sockaddr *)&otherAddr, &len);if(ret < 0){perror("recvfrom error");//perror:interrupt syatem call-->中断系统调用//break;加了break此时会直接跳出while循环,不加则继续进行等待接收(超时检测)}else if(0 == ret){printf("组中停止消息的发送~\n");break;}else{printf("接收的组播消息:%s\n",buf);}}//5,离开多播组setsockopt(socketFd, IPPROTO_IP, IP_DROP_MEMBERSHIP, &ipaddr,sizeof(ipaddr));//6,关闭套接字close(socketFd);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/21385.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android11 AudioTrack 创建过程

Android 系统播放声音&#xff0c;需要创建AudioTrack来和AudioFlinger通信&#xff0c;其创建过程如下 根据传入的声音属性得到output通过得到的output&#xff0c;找到播放线程AudioFlinger在播放线程内&#xff0c;创建Track&#xff0c;和AudioTrack对应。后续通过它们进…

低代码设计中的组织结构的作用与模式

一、组织结构的作用 在低代码设计中&#xff0c;组织结构是系统运作的基石&#xff0c;它定义了系统中的关键元素&#xff0c;包括人员、部门、角色&#xff0c;以及一人多部门、一人多部门多角色的复杂关系。这种定义不仅为系统提供了清晰的运行框架&#xff0c;还确保了系统…

这是一张单纯的图片

说明&#xff1a;通过简单的一张图片找到flag。 打开给出的图片&#xff0c;是一个卡通头像。 方法一&#xff1a; 使用notepad再次打开图片&#xff0c;最后一行发现一行编码。 通过给出的编码格式&#xff0c;可判断是ASCII编码。 使用Ascii编码解码工具&#xff0c;对这串…

贾英才医生:什么是脑白质病?如何预防此病?

贾英才医生&#xff0c;目前就任于北京崇文门中医医院&#xff0c;作为在医学领域有着深厚造诣和丰富经验的专业人士&#xff0c;在此深入探讨和剖析这样一个重要的医学话题&#xff1a;究竟什么是脑白质病&#xff1f;以及我们应该采取怎样切实可行的措施来有效预防此种疾病的…

计算机设备管理器端口不见了怎么办?

因为重装了系统&#xff0c;一切都得重新配置&#xff0c;但突然发现计算机资源设备里面端口这一选项居然不见了&#xff0c;这对一个嵌入式工程师是致命的&#xff0c;赶紧想想怎么处理 1、在电脑左下角windows右键&#xff0c;打开从上往下数第六个“资源管理器” 2、端口没…

Chrome 调试技巧

1. alert 在最早的时候&#xff0c;javascript 程序员调试代码都是通过 alert 进行&#xff0c;但 alert 会让整个程序被打断&#xff0c;并且还有一个很大的缺点&#xff0c;调试完成之后&#xff0c;如果忘记将 alert 删除 or 注释掉&#xff0c;导致别人访问该页面时会莫名…

迅狐短剧小程序源码:打造个性化的追剧体验

随着移动互联网的普及&#xff0c;短剧小程序源码的开发成为了影视爱好者的新宠。它不仅为用户提供了便捷的追剧体验&#xff0c;还通过推荐系统、观看历史、个性化喜好等特色功能&#xff0c;满足了用户的多样化需求。本文将深入探讨短剧小程序源码的特点、优势以及如何实现多…

创新指南|领导者如何评估自己的表现——麦肯锡专有的CEO卓越评估工具

CEO是任何组织中最具挑战性和要求最高的职位之一&#xff0c;尤其是在当前的经济环境下。这也是最重要的职位之一。研究表明&#xff0c;一家公司 45% 的业绩可归因于CEO的影响。但 CEO 们的实际表现如何&#xff1f;他们面临哪些问题&#xff1f;如何帮助他们发挥出最佳水平&a…

贝锐向日葵分组策略:减少重复操作,提升管理效率

面对大数量级的IT设备&#xff0c;如何高效实施管理是运维的关键所在&#xff0c;如何快速准确的对大量的设备按需分组&#xff0c;则是管理精准触达的第一步。 但是&#xff0c;传统的分组方式应付少量设备还可行&#xff0c;设备数量级一旦来到上千台甚至更多时&#xff0c;…

6.4 Go 映射(Map)

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…

Java 异常处理详解(如果想知道Java中有关异常处理的知识点,那么只看这一篇就足够了!)

前言&#xff1a;异常处理是 Java 编程中非常重要的一部分&#xff0c;它能够有效地捕获和处理程序运行中的错误&#xff0c;提高代码的健壮性和可靠性。本文将深入探讨 Java 中异常的概念、体系结构、抛出、解决方式以及如何自定义异常&#xff0c;并结合代码案例进行详细解释…

mysql 如果有按照时间范围查询结果

1.前端传2024-06-21 到我们xml sql 文件中默认实际是 2024-06-21 00:00:00 <if test"dto.startTime ! null">and ps.created_at > #{dto.startTime}</if><if test"dto.endTime ! null">and ps.created_at < #{dto.endTime}</if…

Linux命令篇(六):vi/vim专项

&#x1f49d;&#x1f49d;&#x1f49d;首先&#xff0c;欢迎各位来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里不仅可以有所收获&#xff0c;同时也能感受到一份轻松欢乐的氛围&#xff0c;祝您生活愉快&#xff01; 文章目录 一、什么是vim二…

Microsoft Outlook Lite 引入短信功能

随着科技的不断进步&#xff0c;我们的沟通方式也在不断演变。微软最新推出的 Outlook Lite 应用&#xff0c;不仅为我们提供了一个轻量级的电子邮件管理工具&#xff0c;现在更是带来了一项令人兴奋的新功能——短信服务。 Outlook Lite&#xff1a;轻量级&#xff0c;功能全…

今日好料推荐(ARM嵌入式)

今日好料推荐&#xff08;ARM嵌入式&#xff09; 参考资料在文末获取&#xff0c;关注我&#xff0c;获取优质资源。 给我留言&#xff0c;会帮大家寻找需要的资料。 ARM 嵌入式系统 嵌入式系统在现代电子设备中扮演着至关重要的角色&#xff0c;从智能手机到工业自动化&am…

深入了解 C 语言 Bug

目录 一、引言二、Bug的定义三、Bug的由来四、Bug的影响五、应对 Bug 的方法六、结论 一、引言 1、在 C 语言的编程世界中&#xff0c;Bug 是一个我们无法回避的话题。 2、Bug&#xff0c;简单来说&#xff0c;就是程序中存在的错误或缺陷。它可以表现为程序运行结果的异常、崩…

HALCON-从入门到入门-形态学操作-千变万化的region

1.废话 计算机视觉中的形态学操作是一种基本的数字图像处理技术&#xff0c;主要用于图像的形态结构改变&#xff0c;以便执行前后景分割、形状检测等任务。 膨胀&#xff08;Dilation&#xff09;&#xff1a; 膨胀操作是将结构元素在图像上滑动&#xff0c;如果结构元素与图…

CMakeFile.txt通过sysroot方式后生成makefile报错

报错信息如下&#xff1a; -- The C compiler identification is unknown -- The CXX compiler identification is unknown -- Check for working C compiler: /home/xj/asm/host/bin/aarch64-buildroot-linux-gnu-gcc -- Check for working C compiler: /home/xj/asm/host/bi…

VScode远程连接linux服务器开发,误删了文件怎么找回。

VScode远程连接linux服务器开发&#xff0c;误删了代码文件 因为远程服务器大家都在用&#xff0c;没有足够权限去折腾。找遍了没找到方法&#xff0c;就告诉我远程的文件本地没有缓存啊&#xff01;我就差点开始重写代码了。 后来被我发现了TIMELINE功能&#xff0c;这个功能…

YUM安装httpd实验配置apache

实验目的及实验要求&#xff1a; 实验目的&#xff1a; 2.实验要求&#xff1a; &#xff08;1&#xff09;完成命令的编写&#xff0c;并能正确运行&#xff1b; &#xff08;2&#xff09;从中熟练掌握命令的功能及作用。 实验设备及软件&#xff1a; pc机 配置好Lin…