二十、Linux 进程与信号---非局部跳转
20.1 setjmp 和 longjmp 函数
20.1.1 函数介绍
#include <setjmp.h>
int setjmp(jmp_buf env);
- 函数功能:设置非局部跳转的跳转点(设置跳转点)
- 返回值:直接调用返回0,若从 longjmp 调用返回则返回0
- 这个函数会被执行两次,一次是自己本身使用的时候返回0,另一次再调用 longjump 的时候,此函数再返回 longjmp 中的 val 值
#include <setjmp.h>
void longjmp(jmp_buf env, int val);
- 函数功能:进行非局部跳转,val 为返回值(具体完成跳转,例如goto)
- 参数:
- @env:
- 一个特殊类型 jmp_buf。这一数据类型是某种形式的数组,其中存放在调用 longjmp 时能用来恢复栈状态的所有信息。一般,env 变量是个全局变量,因为需从另一个函数中引用他。
- @env:
- C程序缺乏异常处理的语法,可使用非局部跳转处理C程序的异常
- goto语句仅限于函数内部的跳转,而 longjmp 不限于
20.1.2 例子
process_jmp.c
#include <setjmp.h>
#include <unistd.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h> #define TOK_ADD 5
#define TOK_SUB 6 void do_line(char *line);
void cmd_add(void);
void cmd_sub(void);
int get_token(char *item);/* 获取分割字符 */ char *prompt = "cal:"; /* 命令行提示符 */
jmp_buf env;/* 跳转的 buf 结构 */ int main(void)
{
ssize_t size = strlen(prompt) * sizeof(char);
char buff[];
ssize_t len; /* 设置跳转点 */
/* setjmp 第一次执行成功返回0,调用 longjmp 后此处再返回 非0值 */
26 if(setjmp(env) < 0) {
27 perror("setjmp error");
28 exit(1);
29 }
write(STDOUT_FILENO, prompt, size);
while() {
len = read(STDIN_FILENO, buff, );
if(len < ) break; buff[len - ] = ;
do_line(buff);
write(STDOUT_FILENO, prompt, size);
} return ;
} void do_line(char *line)
{
int cmd = get_token(line); switch(cmd) {
case TOK_ADD:
cmd_add();
break;
case TOK_SUB:
cmd_sub();
break;
default:
fprintf(stderr, "error command\n");
} } void cmd_add(void)
{
int i = get_token(NULL);
int j = get_token(NULL);
printf("result is %d\n", i + j);
} void cmd_sub(void)
{
int i = get_token(NULL);
int j = get_token(NULL);
printf("result is %d\n", i - j);
} static int is_number(char *item)
{
int len = strlen(item);
int i; for(i = ; i < len; i++)
{
if(item[i] > '' || item[i] < '')
return ;
} return ;
} int get_token(char *line)
{
/*
* add 3 4
*/
char *item = strtok(line, " "); if(line != NULL) {
if(!strcmp("add", item)) return TOK_ADD;
if(!strcmp("sub", item)) return TOK_SUB;
} else {
if(is_number(item)) {
int i = atoi(item);
return i;
} else {
fprintf(stderr, "arg not number\n");
/* 如果输入的参数不正常,则让程序跳回到主函数执行下一次循环 */
/* 进行非局部跳转 */
longjmp(env, 1);// 跳转到 setjmp 处执行
}
}
}
执行成功

如果将红色部分注释掉,会发现打印 reasult is xx 数字,xx数字 是一个随机值,因为再 get_token 函数中,fprintf 后就没有做退出也没有做返回实际数字,那么函数运行完毕后,就会返回一个随机值来做加减运行,结果也就变为了一个随机值。
20.2 非局部跳转中,变量的使用
编译器优化编译后:
- 全局变量、静态变量和 volatile(易矢变量)
- 不能恢复到原始值
- 寄存器变量
- 可以恢复到原始值
- 自动变量潜在问题
- 优化编译后可能会恢复
- malloc 变量
- 与编译器优化有关,有的编译器进行优化编译会改变,有的编译器不会,具体看编译器优化
longjmp_val.c
#include <setjmp.h>
#include <stdio.h>
#include <stdlib.h>
#include <malloc.h> int g_val; jmp_buf env; /*
* g_val:全局变量
* s_val:静态变量
* a_val:自动变量,即局部变量
* r_val:寄存器变量
* m_val:通过 malloc 分配的变量
* v_val:易失变量
*/
void fun1(int g_val, int s_val, int a_val, int r_val, int m_val, int v_val); void fun2(); int main(void)
{
static int s_val;
int a_val;
register r_val;
int *m_val = (int *)malloc(sizeof(int));
volatile int v_val; g_val = ;
s_val = ;
a_val = ;
r_val = ;
*m_val = ;
v_val = ; int k = ; if((k = setjmp(env)) < ) {
perror("setjmp error");
exit();
} else if( k == ) {
printf("after longjmp\n");
printf("g_val %d, s_val %d, a_val %d r_val %d, m_val %d v_val %d\n",
g_val, s_val, a_val, r_val, *m_val, v_val);
exit();
} g_val = ;
s_val = ;
a_val = ;
r_val = ;
*m_val = ;
v_val = ; fun1(g_val, s_val, a_val, r_val, *m_val, v_val); return ;
} void fun1(int g_val, int s_val, int a_val, int r_val, int m_val, int v_val)
{
printf("before longjmp\n"); printf("g_val %d, s_val %d, a_val %d r_val %d, m_val %d v_val %d\n",
g_val, s_val, a_val, r_val, m_val, v_val); fun2();
} void fun2()
{
longjmp(env, );
}
不进行优化编译后执行:


优化编译后:


二十、Linux 进程与信号---非局部跳转的更多相关文章
- 三十二、Linux 进程与信号——不可靠信号
32.1 不可靠信号问题 发生信号时关联动作被重置为默认设置 信号可能丢失 程序片段 在进入 sig_int 与再次调用 signal 之间发生的 SIGINT 信号将不会捕获 导致进程终止 以前版本 ...
- 二十二、Linux 进程与信号---进程创建
22.1 fork 和 vfork 函数 22.1.1 函数说明 #include <unistd.h> #include <sys/types.h> pid_t fork( ...
- 二十二、Linux 进程与信号---进程创建(续)
22.2 父子进程操作文件 文件操作由两种模式: IO 系统调用操作文件 标准C IO 操作文件 看代码: #include <unistd.h> #include <string. ...
- Unix系统编程()执行非局部跳转:setjmp和longjmp
使用库函数setjmp和longjmp可执行非局部跳转(local goto). 术语"非局部(nonlocal)"是指跳转目标为当前执行函数之外的某个位置. C语言里面有个&qu ...
- 【转】浅析C语言的非局部跳转:setjmp和longjmp
转自 http://www.cnblogs.com/lienhua34/archive/2012/04/22/2464859.html C语言中有一个goto语句,其可以结合标号实现函数内部的任意跳转 ...
- 三十、Linux 进程与信号——信号的概念及 signal 函数
30.1 信号的基本概念 信号(signal)机制是Linux 系统中最为古老的进程之间的通信机制,解决进程在正常运行过程中被中断的问题,导致进程的处理流程会发生变化 信号是软件中断 信号是异步事件 ...
- 十八、Linux 进程与信号---进程介绍
18.1 进程的概念 程序:程序(program)是存放再磁盘文件中的可执行文件 进程 程序的执行实例被称为进程(process) 一个程序的执行实例可能由多个 进程具有独立的权限和职责.如果系统中某 ...
- 三十四、Linux 进程与信号——信号特点、信号集和信号屏蔽函数
34.1 信号特点 信号的发生是随机的,但信号在何种条件下发生是可预测的 进程杠开始启动时,所有信号的处理方式要么默认,要么忽略:忽略是 SIGUSR1 和 SIGUSR2 两个信号,其他都采取默认方 ...
- 二十三、Linux 进程与信号---进程链和进程扇、守护进程和孤儿进程以及僵尸进程
23.1 进程链和进程扇 23.1.1 概念 进程链:一个父进程构建出一个子进程,子进程再构建出子子进程,子子进程构建出子子子进程.... 这种就为进程链 进程扇:一个父进程构建出多个子进程,子进程都 ...
随机推荐
- 状压DP总结
状态压缩就是将一行的状态压成一个二进制数,这个数的二进制形式反映了这一行的情况 比如0100111的意义为:这一排的第一个数没被使用,第二个被占用了,第三四个没被占用,第五六七个被占用 我们知道位运算 ...
- python安装tesseract
一.最近在学习python爬虫的时候需要用到tesseract,但书上的给的教程对我并不适用,坑了好久天,才终于成功. 二.方法: 1.由于我看的是静谧博主的那本书.他给的教程在python3安装有问 ...
- css 多行文本的溢出显示省略号(移动端)
多行文本的溢出显示省略号(移动端) 一.单行文本的溢出显示省略号(通用) .mui-ellipsis { overflow: hidden; /*规定当文本溢出包含元素时发生的事情*/ white-s ...
- sh: /etc/init.d/sshd: not found Docker中的Alpine镜像安装sshd无法启动
问题描述 在Alpine镜像中安装了openssh-server和openssh之后,无法执行ssh localhost.发现未启动服务,开启服务时报以下错误 / # ls /etc/init.d/s ...
- 【洛谷P1483】序列变换
题目大意:给定一个长度为 N 的序列,有 M 个操作,支持将下标为 x 的倍数的数都加上 y,查询下标为 i 的元素的值. 题解:由于查询操作很少,相对的,修改操作很多.若直接模拟修改操作,即:枚举倍 ...
- 跟我一起学习vue2(熟悉vue.js)[一]
由于有angularjs的基础,所以我第一步是在官网:https://cn.vuejs.org/ 上面看了三遍他的理论知识,还有实例. 现在做完了第二步,就是在菜鸟教程:http://www.runo ...
- Minieye杯第十五届华中科技大学程序设计邀请赛网络赛 部分题目
链接:https://pan.baidu.com/s/12gSzPHEgSNbT5Dl2QqDNpA 提取码:fw39 复制这段内容后打开百度网盘手机App,操作更方便哦 D Grid #inc ...
- 最短路 次短路 k短路(k很小)
最短路 luogu 3371 https://www.luogu.org/problemnew/show/P3371 #include <cstdio> #include <cstd ...
- Day29--Python--缓冲区, 粘包
tcp: 属于长连接,与一个客户端进行连接了以后,其他的客户端要等待.要想连接另外一个客户端,需要优雅地断开当前客户端的连接 允许地址重用:server.setsockopt(socket.SOL_S ...
- redis主从复制配置(1)
我们来配置一个一主两从的服务,根据前面写的已经配置好的redis基础上进行主从配置 一:进入redis的配置目录 cd /usr/local/redis 创建下面3个目录,命令为:make -p /u ...