C语言指针(指针数组和数组指针)

1.指针数组与数组指针

1.1、字面意思来理解指针数组与数组指针
(1)指针数组的实质是一个数组,这个数组中存储的内容全部是指针变量。
(2)数组指针的实质是一个指针,这个指针指向的是一个数组。
1.2、分析指针数组与数组指针的表达式
(1)int *p[5]; int (*p)[5]; int *(p[5]);
(2)一般规律:int *p;(p是一个指针); int p[5];(p是一个数组)
总结:我们在定义一个符号时,关键在于:首先要搞清楚你定义的符号是谁(第一步:找核心);其次再来看谁跟核心最近、谁跟核心结合(第二步:找结合);以后继续向外扩展(第三步:继续向外结合直到整个符号完)。
(3)如何核心和*结合,表示核心是指针;如果核心和[]结合,表示核心是数组;如果核心和()结合,表示核心是函数。
(4)用一般规律来分析3个符号:

第一个:int *p[5]; 核心是p,p是一个数组,数组有5个元素,数组中的元素都是指针,指针指向的元素类型是int类型的;整个符号是一个指针数组。
第二个,int (*p)[5];
核心是p,p是一个指针,指针指向一个数组,数组有5个元素,数组中存的元素是int类型; 总结一下整个符号的意义就是数组指针。
第三个,int *(p[5]); 
解析方法和结论和第一个相同,()在这里是可有可无的。

注意:符号的优先级到底有什么用?其实是决定当2个符号一起作用的时候决定哪个符号先运算,哪个符号后运算。
遇到优先级问题怎么办?第一,查优先级表;第二,自己记住(只要记住[] . ->这几个优先级比较好即可)。

1.3、总结1:优先级和结合性是分析符号意义的关键
(1)在分析C语言问题时不要胡乱去猜测规律,不要总觉得c语言无从捉摸,从已知的规律出发按照既定的规则去做即可。
1.4、总结2:学会逐层剥离的分析方法
(1)找到核心后从内到外逐层的进行结合,结合之后可以把已经结合的部分当成一个整体,再去和整体外面的继续进行结合。
1.5、总结3:基础理论和原则是关键,没有无缘无故的规则

2.函数指针与typedef

2.1、函数指针的实质(还是指针变量)
(1)函数指针的实质是指针,是指针变量。本身占4字节(在32位系统中,所有的指针都是4字节)
(2)函数指针、数组指针、普通指针之间并没有本质区别,区别在于指针指向的东西是个什么玩意。
(3)函数的实质是一段代码,这一段代码在内存中是连续分布的(一个函数的大括号括起来的所有语句将来编译出来生成的可执行程序是连续的),所以对于函数来说很关键的就是函数中的第一句代码的地址,这个地址就是所谓的函数地址,在C语言中用函数名这个符号来表示。
(4)结合函数的实质,函数指针其实就是一个普通变量,这个普通变量的类型是函数指针变量类型,它的值就是某个函数的地址(也就是它的函数名这个符号在编译器中对应的值)

2.2、函数指针的书写和分析方法
(1)C语言本身是强类型语言(每一个变量都有自己的变量类型),编译器可以帮我们做严格的类型检查。
(2)所有的指针变量类型其实本质都是一样的,但是为什么在C语言中要去区分它们,写法不一样呢(譬如int类型指针就写作int *p; 数组指针就写作int (*p)[5],函数指针就得写得更复杂
(3)假设我们有个函数是:void func(void); 对应的函数指针:void (*p)(void); 类型是:void (*)(void);
(4)函数名和数组名最大的区别就是:函数名做右值时加不加&效果和意义都是一样的;但是数组名做右值时加不加&意义就不一样。
(5)写一个复杂的函数指针的实例:譬如函数是strcpy函数(char *strcpy(char *dest, const char *src);),对应的函数指针是:char *(*pFunc)(char *dest, const char *src);

2.3、typedef关键字的用法
(1)typedef是C语言中一个关键字,作用是用来定义(或者叫重命名类型)
(2)C语言中的类型一共有2种:一种是编译器定义的原生类型(基础数据类型,如int、double之类的);第二种是用户自定义类型,不是语言自带的是程序员自己定义的(譬如数组类型、结构体类型、函数类型·····)。
(3)我们今天讲的数组指针、指针数组、函数指针等都属于用户自定义类型。
(4)有时候自定义类型太长了,用起来不方便,所以用typedef给它重命名一个短点的名字。
(5)注意:typedef是给类型重命名,也就是说typedef加工出来的都是类型,而不是变量。
练习题目:函数指针与typedef的使用

#include <stdio.h>
#include <string.h>

void func1(void)
{
	printf("I am func1.\n");
}

// 这句重命名了一种类型,这个新类型名字叫pType,类型是:char* (*)(char *, const char *);
typedef char* (*pType)(char *, const char *);

// 函数指针数组
typedef char* (*pType[5])(char *, const char *);
// 函数指针数组指针
typedef char* (*(*pType)[5])(char *, const char *);


int main(void)
{
	char* (*p1)(char *, const char *);
	char* (*p2)(char *, const char *);
	
	pType p3;		// 等效于 char* (*p3)(char *, const char *);
	pType p4;
	
	p3 = p1;
	
/*
	char a[5] = {0};
	char* (*pFunc)(char *, const char *);
	pFunc = strcpy;
	pFunc(a, "abc");
	printf("a = %s.\n", a);
*/

	
/*	
	void (*pFunc)(void);
	//pFunc = func1;			// 左边是一个函数指针变量,右边是一个函数名
	pFunc = &func1;				// &func1和func1做右值时是一模一样的,没任何区别
	pFunc();				    // 用函数指针来解引用以调用该函数
*/
	
/*	
	int *p;
	int a[5];
	
	p = a;		    // 匹配的,类型匹配的,所以编译器不会警告不会报错。
	//p = &a;		// 类型不匹配,p是int *, &a是int (*)[5];
	
	int (*p1)[5] ;
	p1 = &a;		// p1类型是int (*)[5],&a的类型也是int (*)[5]
*/	
	return 0;
}

2.4、总结:函数指针的分析方法也是源于优先级与逐层剥离的基本理论

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值