一、请大家看一个有趣的题目(问输出结果是什么?)
//上代码
#include<stdio.h>
int main(){
const char* str[]={"hello","xiaoxin","printf","scanf","xiaobai"};
const char* s=str[1];
int n=sizeof(str)/sizeof(str[0]);
for(int i=0;i<n;i++){
printf("%s \n",s);
s=s+1;
}
return 0;
}
大家思考完后,我先放上程序运行结果截图:
哈哈,接下来请看解析(全标在代码注释中!)
int main(){
const char* str[]={"hello","xiaoxin","printf","scanf","xiaobai"}; //字符型指针数组,每个元素为一个字符型指针,指向其对应字符串首地址(首元素地址)
const char* s=str[1]; //s是一个字符型指针,指向 "xiaoxin" 这个字符串
int n=sizeof(str)/sizeof(str[0]); //n=str[]数组的元素个数=5
for(int i=0;i<n;i++){
printf("%s \n",s); //i=0 s指向"xiaoxin"首字母地址 输出字符串:xiaoxin i=1 s指向"xiaoxin"第二个字母地址 输出字符串:iaoxin 以此类推...
s=s+1;
}
return 0;
}
针对s=s+1还是想多说几句:s是一个字符型指针 指向字符型指针数组str[]的第二个元素(下标为1) 等价于可以将s想成是一个字符数组(char s[]={"xiaoxin"};),s本来指向数组首元素地址,s+1后指向数组第二个元素地址,再对应输出字符串即可。这样应该会比较好理解。(除了sizeof(str)、&str 中数组名代表整个数组,其余地方出现的数组名都代表数组首元素地址!)
二、分析下列定义的是数组?指针?(答案在注释中)
判断方法:从右向左分析!!先解释(最里面的)括号!!
int *p1[10]; //从右向左解释,p1为一个数组!有10个元素,每个元素为一个整型指针!
int (*p2)[10]; //p2为一个指针!指向一个有10个元素的整型数组(存放数组的地址)
int *(*p3)[10]; //p3为一个指针!指向一个有10个元素的整型指针数组
int *fun(int a,int b); //fun为一个拥有2个整型形参的函数,返回类型为:int*
int(*pfun)(int a,int b); //pfun为一个指针!指向一个拥有2个整型形参的返回类型为整型的函数
int *(*pfun2)(int a,int b); //pfun2为一个指针!指向一个拥有2个整型形参的返回类型为int*的函数
int (*pfun[3])(int a,int b); //pfun为一个数组!有三个元素,每个元素为指针,存放函数的地址
//举例:
int ar[10];
int (*pr)[10]=&ar;
int *br[10];
int *(*pr)[10]=&br;
三、形参入栈规则
对于C语言和C++,形参入栈均是从右向左依次入栈!!(java相反)
#include<stdio.h>
void fun(int a){
int *p=&a;
printf("%d\n",*p); //输出12
p++; //指向栈中下一个元素23
printf("%d\n",*p); //输出23
p++;
printf("%d\n",*p); //输出34
p++;
}
int main(){
fun(12,23,34); //使用Vs2019编译环境ok
//形参入栈规则:从右向左依次入栈!!
return 0;
}