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的次数也有关系

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

相关推荐
白緢7 分钟前
嵌入式 Linux + 内核开发高频问题及排查
java·linux·运维
学编程就要猛8 分钟前
JavaEE初阶:网络编程
运维·服务器·网络
Proxy_ZZ010 分钟前
打造自己的信道编码工具箱——Turbo、LDPC、极化码三合一
c语言·算法·信息与通信
hughnz13 分钟前
钻井自动化案例研究
运维·自动化
ILL11IIL17 分钟前
Docker容器技术
运维·docker·容器
蜡笔小新..19 分钟前
Linux下Matplotlib使用Times New Roman字体的解决方案
linux·运维·matplotlib
资深流水灯工程师20 分钟前
FREERTOS的核心内容与核心组件
笔记
飞yu流星21 分钟前
文件压缩、文本内容、文本编辑
运维·服务器
郭泽斌之心22 分钟前
Fay 的多用户对话消息分发逻辑
经验分享·fay数字人
无限进步_23 分钟前
【C++】重载、重写和重定义的区别详解
c语言·开发语言·c++·ide·windows·git·github