linux在使用重定向写入文件时(使用标准C库函数时)使处理信号异常(延时)--问题分析

linux在使用重定向写入文件时(使用标准C库函数时)使处理信号异常(延时)--问题分析

在使用alarm函数进行序号处理测试的时候发现如果把输出重定向到文件里面会导致信号的处理出现严重的延迟(ubuntu18)

c 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
int main(void){
	alarm(3);//定时三秒
	while(1){
		int i;
		printf("%d\n", i++); //打印信息
	}
	return 0;
}

测试

  • 正常情况

使用命令time ./a.out进行计时

这时候这是个时间是基本没有误差的

  • 出现问题

这时候使用命令time ./a.out > text.out把输出的信息放到文件里面

这时候就会发现这一个处理的时间开始不对劲了

尝试分析

库函数分析

c 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
int main(void){
	alarm(3);
	printf("now alarm  = %d\n", alarm(3));
	while(1){
		char buf[20];
		int i;
		sprintf(buf, "%d\n", i++);
		write(STDOUT_FILENO, buf, strlen(buf));
	}
	return 0;
}

把这一个输出使用系统调用进行

这里可以看出延时少了很多

查看实际的系统调用

使用命令strace ./a.out > out.txt查看实际的系统调用

  • 库函数版本(shell)

  • 库函数版本(文件)

  • 系统调用版本(shell)

  • 系统调用版本(文件)

这时候可以发现这库函数进行文件写入的时候是使用了一个缓存区4096字节

测试缓冲区大小对信号处理的影响

c 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
int main(void){
	alarm(3);
	printf("now alarm  = %d\n", alarm(3));
	while(1){
		int i;
		printf("%d ", i++);//不使用换行, 这时候库函数会使用大的缓冲区
	}
	return 0;
}


这时候可以获取结论这一个不是因为缓冲区大小出现的, 写入shell的时候使用的是1024的缓冲区, 但是对于时间的影响不大

write函数分析

在man文档里面的write里面有这样一段话

If a write() is interrupted by a signal handler before any bytes are written, then the call fails with the error EINTR; if it is interrupted after at least one byte has been written, the call succeeds, and returns the number of bytes written.

从这一段可以推测出write不是原子操作, 并且这一个是可以被信号打断的, 并且实际测试写shell的时候是随时可以使用Ctrl + C打断的, 只有在文件处理的时候会出现Ctrl + C无法立刻打断这一个程序

信号处理分析

信号处理分析

从这一篇文章里面可以获取信号处理是在从内核态返回的时候处理的

推测结论

猜测可能是内核在对文件处理的时候会把很多的write里面的信息合并起来处理(就算是4096处理一次从跟踪可以看出来也是有很多次的, 如果每一处只处理一个4096不至于出现延时达到好几秒的情况), 写入shell的时候没有进行合并, 所以误差比较小, 这时候由于硬盘的写入比较慢, 数据量比较大, 需要等待IO, 会长时间处于内核态, 导致信号处理的延时

使用write每一次写入的数量比较少的时候反应比较快, 推测这一个合并可能是和write的次数也有关系

这一段仅供参考, 没有实际查看源码, 回头分析源码以后会再补充的

相关推荐
li99yo2 小时前
3DGS的复现
图像处理·pytorch·经验分享·python·3d·conda·pip
浅念-4 小时前
Linux 开发环境与工具链
linux·运维·服务器·数据结构·c++·经验分享
2501_926978335 小时前
AI的三次起落发展分析,及未来预测----理论5.0的应用
人工智能·经验分享·笔记·ai写作·agi
潜创微科技--高清音视频芯片方案开发6 小时前
2026年C转DP芯片方案深度分析:从适配场景到成本性能的优选指南
c语言·开发语言
似水এ᭄往昔6 小时前
【Linux】gdb的使用
linux·运维·服务器
优雅的造轮狮6 小时前
WSL2 Docker Desktop配置优化及迁移D盘指南
运维·docker·容器
tian_jiangnan6 小时前
grafana白皮书
linux·服务器·grafana
枫桥骤雨6 小时前
我的龙虾日记
ubuntu·openclaw
大师影视解说6 小时前
基于Web端的AI电影解说自动化生产工具实测:4步完成从文案到成片的全流程
运维·人工智能·自动化·影视解说·电影解说工具·网页版电影解说·ai电影解说
路小雨~7 小时前
Transformer架构学习笔记:从数学推导到工程实现与主流变体
笔记·ai·transformer