1 基本功能

本Blog创建了两个进程(父子进程):

  • 父进程

    执行文本复制操作,当收到 SIGUSR1信号后,打印出现在文件复制的进度;
  • 子进程

    每个固定时间段向父进程发送一个 SIGUSR1 信号。

2 代码示例

/*
* File: Signal.c
*Description: Two process
1. Father: copy a file, when receive the SIGUSR1 signal, print the progress
2. Child: timing trigger the parent process
*Autor: Jimmy Nie
*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <signal.h>
#include <unistd.h>
#include <errno.h>
#include <sys/fcntl.h> void handler(int sig);
void SigAlarm(int sig); int count = 0; //has read bytes
int fileSize = 0; //the source file size int main(int argc, char *argv[])
{
//variable define
int fd_src, fd_dst;
int tmp = 0; //how many bytes read every time
char buf[128] ; //tempature storage buffer //0. check the argument
if(argc != 3)
{
printf("%s(%d): Check the arguments, argument=%d(3 is need)\n", argc);
exit(EXIT_FAILURE);
} //1. open the file
if(-1 == (fd_src=open(argv[1], O_RDONLY)))
{
perror("open");
exit(EXIT_FAILURE);
} //open the destination file, if it does not exist, creat it first
if(-1 == (fd_dst=open(argv[2], O_RDWR|O_CREAT, 0644)))
{
perror("open");
exit(EXIT_FAILURE);
} //2. Obtain the source file size
fileSize = lseek(fd_src, 0, SEEK_END);
if(fileSize < 0)
{
perror("lseek");
exit(EXIT_FAILURE);
} lseek(fd_src, 0, SEEK_SET); //3. Father process install SIGUSR1 signal
if(signal(SIGUSR1, handler) == SIG_ERR)
{
perror("signal");
exit(EXIT_FAILURE);
} //4. Creat a new process(child)
pid_t pid;
if(-1 == (pid=fork()))
{
perror("fork");
exit(EXIT_FAILURE);
} //In child process
else if(pid == 0)
{
//Install the signal SIGALRM
if(signal(SIGALRM, SigAlarm) == SIG_ERR)
{
perror("signal");
exit(EXIT_FAILURE);
} ualarm(200,10000); //after 20ms start trigger, and every 50ms trigger once
//alarm(1); while(1) //execute continues
;
} //In parent process
else
{
//3. copy source file to destination file
while(1)
{
//read the source file to buf
if(-1 == (tmp=(read(fd_src, buf, 128))))
{
perror("read");
exit(EXIT_FAILURE);
} //check the end of file
if(0 == tmp)
{
printf("Finished copy the file, and file size:%d\n", fileSize);
kill(pid, SIGINT); //finished copy, trigger a signal to child, and terminate child process
break;
} //write the buffer to the destination file
if(-1 == write(fd_dst, buf, tmp))
{
perror("Write");
exit(EXIT_FAILURE);
} count += tmp;
} wait(NULL); //wait child process exit
close(fd_src);
close(fd_dst);
} return 0;
} //function used to print the progree of copy file
void handler(int sig)
{
int i = 0;
i = (int)(((float)count / (float)fileSize) * 100);
printf("\nHas copyed %d%% \n",i); int j = 0;
for(j=0; j<i; j++)
{
if(j%2)
printf("*");
} printf("\n");
} //used to send SIGUSR1 signal to parent process
void SigAlarm(int sig)
{
kill(getppid(), SIGUSR1);
//alarm(1);
}

编译该代码:


[root@niesh Linux]# gcc -o signal signal.c
[root@niesh Linux]# ll
总用量 36
-rwxrwxr-x. 1 niesh niesh 8659 9月 22 22:07 produce
-rw-rw-r--. 1 root niesh 467 9月 22 22:07 produce.c
-rwxr-xr-x. 1 root root 13445 9月 23 11:26 signal
-rw-rw-r--. 1 root niesh 3407 9月 22 22:45 signal.c

但是此时我们还需要一个大于1M的ASCII文件,使得CP不至于瞬间完成:

#include <stdio.h>
#include <sys/fcntl.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h> int main(int argc, char *argv[])
{
int fd;
int count = 0;
char buf[] = "Hello,world\n"; fd = open(argv[1], O_RDWR|O_CREAT, 0644); for(count=0; count < 1024*1024; count++)
write(fd, buf, strlen(buf)); //file size = 16*256*1024*1024 = 4Mbyte //printf("The sizeof(buf)=%d\n",sizeof(buf));
close(fd); return 0;
}

通过以上代码我们可以产生出一个12M的文本文件:


[root@niesh Linux]# gcc -o produce produce.c
[root@niesh Linux]# ./produce test
[root@niesh Linux]# ll -h
总用量 13M
-rwxr-xr-x. 1 root root 8.5K 9月 23 11:31 produce
-rw-rw-r--. 1 root niesh 467 9月 22 22:07 produce.c
-rwxr-xr-x. 1 root root 14K 9月 23 11:26 signal
-rw-rw-r--. 1 root niesh 3.4K 9月 22 22:45 signal.c
-rw-r--r--. 1 root root 12M 9月 23 11:31 test //ASCII文件 12M

3 执行效果


[root@niesh Linux]# ./signal test t1 Has copyed 1% Has copyed 3%
* Has copyed 6%
*** Has copyed 7%
*** Has copyed 9%
**** Has copyed 10%
***** Has copyed 15%
******* Has copyed 21%
********** Has copyed 25%
************ Has copyed 29%
************** Has copyed 34%
***************** Has copyed 42%
********************* Has copyed 44%
********************** Has copyed 48%
************************ Has copyed 54%
*************************** Has copyed 58%
***************************** Has copyed 65%
******************************** Has copyed 71%
*********************************** Has copyed 76%
************************************** Has copyed 80%
**************************************** Has copyed 88%
******************************************** Has copyed 93%
********************************************** Has copyed 97%
************************************************
Finished copy the file, and file size:12582912

查看复制后的结果:


[root@niesh Linux]# ll -h
总用量 25M
-rwxr-xr-x. 1 root root 8.5K 9月 23 11:31 produce
-rw-rw-r--. 1 root niesh 467 9月 22 22:07 produce.c
-rwxr-xr-x. 1 root root 14K 9月 23 11:26 signal
-rw-rw-r--. 1 root niesh 3.4K 9月 22 22:45 signal.c
-rw-r--r--. 1 root root 12M 9月 23 11:42 t1 //复制后生成的文件
-rw-r--r--. 1 root root 12M 9月 23 11:31 test

Linux信号机制代码示例的更多相关文章

  1. Linux信号机制

    Linux信号(signal) 机制分析 [摘要]本文分析了Linux内核对于信号的实现机制和应用层的相关处理.首先介绍了软中断信号的本质及信号的两种不同分类方法尤其是不可靠信号的原理.接着分析了内核 ...

  2. 利用linux信号机制调试段错误(Segment fault)

    在实际开发过程中,大家可能会遇到段错误的问题,虽然是个老问题,但是其带来的隐患是极大的,只要出现一次,程序立即崩溃中止.如果程序运行在PC中,segment fault的调试相对比较方便,因为可以通过 ...

  3. 利用linux信号机制调试段错误(Segment fault)【转】

    转自:http://blog.csdn.net/ab198604/article/details/6164517 版权声明:本文为博主原创文章,未经博主允许不得转载. 在实际开发过程中,大家可能会遇到 ...

  4. linux信号机制与python信号量

    1.信号本质 软中断信号(signal,又简称为信号)用来通知进程发生了异步事件.在软件层次上是对中断机制的一种模拟,在原理上,一个进程收到一个信号与处理器收到一个中断请求可以说是一样的.信号是进程间 ...

  5. xenomai内核解析之信号signal(一)---Linux信号机制

    版权声明:本文为本文为博主原创文章,转载请注明出处.如有错误,欢迎指正.博客地址:https://www.cnblogs.com/wsg1100/ 目录 1. Linux信号 1.1注册信号处理函数 ...

  6. linux信号机制 - 用户堆栈和内核堆栈的变化【转】

    转自:http://itindex.net/detail/16418-linux-%E4%BF%A1%E5%8F%B7-%E5%A0%86%E6%A0%88 此文只简单分析发送信号给用户程序后,用户堆 ...

  7. linux 信号机制

    文章目录 1. 实时信号非实时信号 2. 信号状态: 3. 信号生命周期: 4. 信号的执行和注销 信号掩码和信号处理函数的继承 信号处理函数的继承 信号掩码的继承 sigwait 与多线程 sigw ...

  8. Storm入门(七)可靠性机制代码示例

    一.关联代码 使用maven,代码如下. pom.xml  参考 http://www.cnblogs.com/hd3013779515/p/6970551.html MessageTopology. ...

  9. Linux信号(signal) 机制分析

    Linux信号(signal) 机制分析 [摘要]本文分析了Linux内核对于信号的实现机制和应用层的相关处理.首先介绍了软中断信号的本质及信号的两种不同分类方法尤其是不可靠信号的原理.接着分析了内核 ...

随机推荐

  1. 【转一篇出处不明的文章】 Windows多线程通信方式

    多线程通信的方法主要有以下三种: 1.全局变量 进程中的线程间内存共享,这是比较常用的通信方式和交互方式.注:定义全局变量时最好使用volatile来定义,以防编译器对此变量进行优化. 2.Messa ...

  2. Redis Sentinel基本介绍(翻译以及总结)

    目录 Redis Sentinel介绍 分布式的Redis Sentinel 快速开始 获取Sentinel 启动Sentinel 部署Sentinel的基本要求 配置Sentinel 其他的Sent ...

  3. RabbitMQ 消息队列 DEMO

    1. 引用 RabbitMQ.Client.5.1.0 2. http://localhost:15672/ public class TestController : ApiController { ...

  4. WPF制作表示透明区域的马赛克画刷

    最近在用WPF制作一款软件,需要像ps一样表示透明区域,于是制作了一个马赛克背景的style.实现比较简单,那么过程和思路就不表了,直接上代码 <DrawingBrush TileMode=&q ...

  5. FunDA(5)- Reactive Streams:Play with Iteratees

    FunDA的设计目标就是把后台数据库中的数据搬到内存里,然后进行包括并行运算的数据处理,最后可能再对后台数据库进行更新.如果需要把数据搬到内存的话,那我们就必须考虑内存是否能一次性容纳所有的数据,有必 ...

  6. spring json的支持

    在spring中可以通过配置来实现对json的支持: 以下连接是看到的一篇对这方面内容讲解比较好的文章 http://www.cnblogs.com/fangjian0423/p/springMVC- ...

  7. jvm(1)类的加载(二)(自定义类加载器)

    [深入Java虚拟机]之四:类加载机制 1,从Java虚拟机的角度,只存在两种不同的类加载器: 1,启动类加载器:它使用C++实现(这里仅限于Hotspot,也就是JDK1.5之后默认的虚拟机,有其他 ...

  8. 【9】JMicro微服务-发布订阅消息服务

    如非授权,禁止用于商业用途,转载请注明出处作者:mynewworldyyl 1. JMicro消息服务目前实现特性 a. JMicro只支持发布订阅消息服务,不支持队列式消息服务: b. 不支持消息持 ...

  9. POJ 2339

    #include <iostream> #include <algorithm> #define MAXN 205 using namespace std; char _m[M ...

  10. JavaScript 函数定义方法

    JavaScript 函数定义方法. 函数声明 在之前的教程中,你已经了解了函数声明的语法 : function functionName(parameters) { 执行的代码 } 函数声明后不会立 ...