goto不可以跨函数跳转,函数setjmp 和 longjmp 可以。
正常调用代码: main-> a -> b -> c
#include<stdio.h>
#include<stdlib.h>
void c(void)
{
printf("%s:begin\n",__FUNCTION__);
printf("%s:end\n",__FUNCTION__);
}
void b(void)
{
printf("%s:begin\n",__FUNCTION__);
printf("%s:call c\n",__FUNCTION__);
c();
printf("%s:return c\n",__FUNCTION__);
printf("%s:end\n",__FUNCTION__);
}
void a(void)
{
printf("%s:begin\n",__FUNCTION__);
printf("%s:call b\n",__FUNCTION__);
b();
printf("%s:return b\n",__FUNCTION__);
printf("%s:end\n",__FUNCTION__);
}
int main(int argc,char** argv)
{
printf("%s:begin\n",__FUNCTION__);
printf("%s:call a\n",__FUNCTION__);
a();
printf("%s:return a\n",__FUNCTION__);
printf("%s:end\n",__FUNCTION__);
exit(0);
}
结果:
main:begin
main:call a
a:begin
a:call b
b:begin
b:call c
c:begin
c:end
b:return c
b:end
a:return b
a:end
main:return a
main:end
运用setjmp和longjmp函数从c函数跳到main并带回数字6:
#include<stdio.h>
#include<stdlib.h>
#include<setjmp.h>
jmp_buf save;
void c(void)
{
printf("%s:begin\n",__FUNCTION__);
printf("%s jump!!!\n",__FUNCTION__);
longjmp(save,6);
printf("%s:end\n",__FUNCTION__);
}
void b(void)
{
printf("%s:begin\n",__FUNCTION__);
printf("%s:call c\n",__FUNCTION__);
c();
printf("%s:return c\n",__FUNCTION__);
printf("%s:end\n",__FUNCTION__);
}
void a(void)
{
printf("%s:begin\n",__FUNCTION__);
printf("%s:call b\n",__FUNCTION__);
b();
printf("%s:return b\n",__FUNCTION__);
printf("%s:end\n",__FUNCTION__);
}
int ret;
int main(int argc,char** argv)
{
printf("%s:begin\n",__FUNCTION__);
ret=setjmp(save);
if(ret==0)
{
printf("%s:call a\n",__FUNCTION__);
a();
printf("%s:return a\n",__FUNCTION__);
}
else
{
printf("%s jump back with %d\n",__FUNCTION__,ret);
}
printf("%s:end\n",__FUNCTION__);
exit(0);
}
结果:对比可知返回时的打印都不打印了
main:begin
main:call a
a:begin
a:call b
b:begin
b:call c
c:begin
c jump!!!
main jump back with 6
main:end
当longjmp函数传回0时,系统会自己用1替代,即带回的ret为1.