你看得懂Hello world吗?
代码如下:#include<stdio.h>
char *a(){return "H";}
char *a1(){return "e";}
char *a2(){return "l";}
char *a3(){return "l";}
char *a4(){return "o";}
char *a5(){return " ";}
char *a6(){return "W";}
char *a7(){return "o";}
char *a8(){return "r";}
char *a9(){return "l";}
char *a10(){return "d";}
char *a11(){return "!";}
int main()
{ int i;
char *(*m)()={
{a,a1,a2,a3},
{a4,a5,a6,a7},
{a8,a9,a10,a11}
};
char*(**p)();
p=&m;//这里用p=m;为啥是一样的结果?
for(i=0;i<12;i++)
{
printf("%s",(*(p+i))());
}
printf("\n");
return 0;
}//解释一下这个程序是怎么实现的?
这段程序使用了函数指针和二维数组来实现输出字符串 "Hello World!"。下面对代码进行逐行解释:
```c
#include<stdio.h>
头文件包含标准输入输出库。
char *a(){return "H";}
char *a1(){return "e";}
char *a2(){return "l";}
char *a3(){return "l";}
char *a4(){return "o";}
char *a5(){return " ";}
char *a6(){return "W";}
char *a7(){return "o";}
char *a8(){return "r";}
char *a9(){return "l";}
char *a10(){return "d";}
char *a11(){return "!";}
定义了12个返回类型为 char* 的函数,每个函数返回的是一个字符。
int main()
{
int i;
char *(*m)()={
{a,a1,a2,a3},
{a4,a5,a6,a7},
{a8,a9,a10,a11}
};
char*(**p)();
p=&m;//这里用p=m;为啥是一样的结果?
for(i=0;i<12;i++)
{
printf("%s",(*(p+i))());
}
printf("\n");
return 0;
}
```
main 函数中定义了一个二维函数指针数组 m,其中每个元素都是一个指向一个返回类型为 char* 的函数的指针。接着定义了一个指向函数指针的指针 p,并将其指向数组 m。然后通过循环执行每个函数,并将函数返回的字符打印出来,最终输出 "Hello World!"。
至于为什么 `p=&m;` 和 `p=m;` 的结果是一样的,这是因为数组名本身就是一个指向数组首元素的指针,所以它和 `&m` 是等价的。 1、程序定义了12个函数(a, a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, a11),分别返回 "Hello World!" 字符串中的每个字符。
2、在main函数中,程序创建了一个3x4的二维数组m,它包含指向这些函数的指针。数组的结构如下:
m - {a,a1,a2,a3}
m - {a4, a5,a6,a7}
m - {a8, a9, a10, a11}
3、定义一个指针(char *(**p)()),它指向这个二维数组的指针。
4、为p赋值(p=&m;),使其指向m。在这个例子中,p=&m;和p=m;可以得到相同的结果,因为m是一个指向指针数组的指针,&m也是一个指向指针数组的指针。
5、程序使用一个for循环来遍历这个二维数组中的所有函数指针,并通过解引用(dereferencing)这些函数指针来调用它们,将返回的字符依次打印出来。最后,循环结束后打印一个换行符。
代码执行后的输出是 "Hello World!"。 {:10_256:}还没看到比较准确的答案 年,兮 发表于 2023-4-5 15:25
还没看到比较准确的答案
{:5_109:}
0x401488 push %ebp
0x401489 mov %esp,%ebp
0x40148b and $0xfffffff0,%esp
0x40148e sub $0x50,%esp
0x401491 call 0x401b60 <__main>
0x401496 movl $0x401410,0x18(%esp)
0x40149e movl $0x40141a,0x1c(%esp)
0x4014a6 movl $0x401424,0x20(%esp)
0x4014ae movl $0x40142e,0x24(%esp)
0x4014b6 movl $0x401438,0x28(%esp)
0x4014be movl $0x401442,0x2c(%esp)
0x4014c6 movl $0x40144c,0x30(%esp)
0x4014ce movl $0x401456,0x34(%esp)
0x4014d6 movl $0x401460,0x38(%esp)
0x4014de movl $0x40146a,0x3c(%esp)
0x4014e6 movl $0x401474,0x40(%esp)
0x4014ee movl $0x40147e,0x44(%esp)
0x4014f6 lea 0x18(%esp),%eax
0x4014fa mov %eax,0x48(%esp)
0x4014fe movl $0x0,0x4c(%esp)
0x401506 jmp 0x401532 <main+170>
0x401508 mov 0x4c(%esp),%eax
0x40150c lea 0x0(,%eax,4),%edx
0x401513 mov 0x48(%esp),%eax
0x401517 add %edx,%eax
0x401519 mov (%eax),%eax
0x40151b call *%eax
0x40151d mov %eax,0x4(%esp)
0x401521 movl $0x406056,(%esp)
0x401528 call 0x403fbc <printf>
0x40152d addl $0x1,0x4c(%esp)
0x401532 cmpl $0xb,0x4c(%esp)
0x401537 jle 0x401508 <main+128>
0x401539 movl $0xa,(%esp)
0x401540 call 0x403fb4 <putchar>
0x401545 mov $0x0,%eax
0x40154a leave
0x40154b ret
jhq999 发表于 2023-4-5 15:50
你是魔鬼吧{:10_277:} 年,兮 发表于 2023-4-5 18:07
你是魔鬼吧
{:5_109:}
其实就是函数指针的3X4数组,然后挨个调用这些函数
调用函数本来就是压栈跳转到函数所在的内存地址 jhq999 发表于 2023-4-5 18:18
其实就是函数指针的3X4数组,然后挨个调用这些函数
调用函数本来就是压栈跳转到函数所在的内 ...
{:5_97:}你是懂底层代码的
页:
[1]