五、附带数据信息的信号处理
int sigaction(int signum, const struct sigaction *act,struct sigaction *oldact);
功能:向内核注册一个信号处理函数
signum:要捕获的信号编号
act: 设置要处理的动作
olact: 获取原来的处理动作
struct sigaction {
void (*sa_handler)(int);// 不附带其他信息的信号处理函数
void (*sa_sigaction)(int,siginfo_t *,void *);
// 附带其他信息的信号处理函数
sigset_t sa_mask; // 信号屏蔽集
int sa_flags;// 信号处理动作标志
SA_NODEFER 在信号处理过程中不要屏蔽当前信号
SA_SIGINFO 使用 sa_sigaction 函数指针
void (*sa_restorer)(void); // 保留函数
};
siginfo_t {
int si_signo; /* Signal number */
int si_errno; /* An errno value */
int si_code; /* Signal code */
int si_trapno;
pid_t si_pid; /* Sending process ID */
uid_t si_uid; /* Real user ID of sending process */
int si_status; /* Exit value or signal */
clock_t si_utime; /* User time consumed */
clock_t si_stime; /* System time consumed */
sigval_t si_value; /* Signal value */
int si_int; /* POSIX.1b signal */
void *si_ptr; /* POSIX.1b signal */
int si_overrun; /* Timer overrun count;
POSIX.1b timers */
int si_timerid; /* Timer ID; POSIX.1b timers */
void *si_addr; /* Memory location which caused fault */
long si_band; /* Band event (was int in
glibc 2.3.2 and earlier) */
int si_fd; /* File descriptor */
short si_addr_lsb; /* Least significant bit of address
(since Linux 2.6.32) */
void *si_call_addr; /* Address of system call instruction
(since Linux 3.5) */
int si_syscall; /* Number of attempted system call
(since Linux 3.5) */
unsigned int si_arch; /* Architecture of attempted system call
(since Linux 3.5) */
}
int sigqueue(pid_t pid,int sig, const union sigval value);
功能:向指定进程发送信号并附加信息一起发送
value:附加的信息
union sigval {
int sival_int; // 整数
void *sival_ptr; // 指针
};
测试1.
使用sigaction函数(携带信息),向内核注册一个信号处理函数
action.sa_sigaction=sigint; 调用sigint函数
执行结果如下
六、定时器
#include <sys/time.h>
int getitimer(int which,struct itimerval *curr_value);
功能:获取当前的定时方案
which:选择使用的计时器
ITIMER_REAL 真实计时器 程序总的计时时间
ITIMER_VIRTUAL 虚拟计时器 用户态的计时
ITIMER_PROF 实际计时器 用户态+内核态计时
真实计时器 = 实际计时器 + 休眠时间 + 切换时间
int setitimer(int which, const struct itimerval *new_value,struct itimerval *old_value);
功能:设置定时器计时方案
which:选择使用的计时器
ITIMER_REAL 真实计时器 程序总的计时时间SIGALRM(14)
ITIMER_VIRTUAL 虚拟计时器 用户态的计时SIGVTALRM(26)**使用的时候要对应
ITIMER_PROF 实际计时器 用户态+内核态计时
SIGPROF(27)
struct itimerval {
struct timeval it_interval;
// 每次时钟信号产生的间隔时间
struct timeval it_value;
// 第一次产生时钟信号的时间 **3秒钟一次,每个两秒
}; *
struct timeval {
time_t tv_sec; // 设置秒3
suseconds_t tv_usec; // 设置微秒50000
};
测试2
结果如下:3秒钟一次,每个两秒
二、创建进程
pid_t fork(void);
功能:创建子进程
返回值:一次调用两次返回,子进程返回0,父进程返回子进程的ID,当进程数量超过系统的限制时会创建失败,返回-1
1、通过fork创建的子进程会拷贝父进程(数据段、bss段、堆、栈、I/O缓冲区),与父进程共享代码段、子进程会继承父进程的信号处理方式
2、fork函数调用后父子进程各自独立运行,谁先返回不确定,但是可以通过睡眠确定让哪个进程先执行
3、通过fork创建的子进程可以共享父进程的文件描述符
4、可以根据返回值的不同让父子进程进入不同的分支,执行不同的代码
测试3.
fork()创建子进程,但那个先执行完是随机的
4、可以根据返回值的不同让父子进程进入不同的分支,执行不同的代码
结果如下:
测试4.
预想会等sleep(1),后父进程,子进程都执行后执行
2、fork函数调用后父子进程各自独立运行,谁先返回不确定,但是可以通过睡眠确定让哪个进程先执行
测试结果如下:
c
测试5.
一分为二:就是拷贝了一份父程序的。
fork()拷贝了父程序,缓冲区也被拷贝了,if加一个\n就只会打印一个
1.通过fork创建的子进程会拷贝父进程(数据段、bss段、堆、栈、I/O缓冲区),与父进程共享代码段
测试结果如下:相当于1分为8:
测试6.
子进程会继承父进程的信号处理方式
结果如下
按下cv键,会执行信号输出,但子进程也会进行信号输出
测试7.
3、通过fork创建的子进程可以共享父进程的文件描述符
测试结果如下:
成功写入文件:test.txt