1.指针与数组(一维数组、二维数组)
int a[10];
int *p=a;
*(a+i), a[i], *(p+i), p[i]
字符数组的值:
1.定义的同时初始化
2.输入函数scanf\gets\fgets
3.字符串函数strcpy
char a[]="hello world";
char *p=a;
p="today"; 对 把内存中“today”字符串常量的地址赋值给了p
a="today" 错
结论:指针是地址变量,而数组名是地址常量,数组名在运算中可以作为指针参与,但不允许被赋值,指针可变,允许指向别的位置
指针和数组常见等价操作:
指针操作 数组操作 说明
p &a[0] 数组首地址
*p a[0] 数组的第一个元素
p+i &a[i] 数组的第i个元素的地址
*(p+i) a[i] 数组的第i个元素
*p+b a[0]+b 数组的第一个元素的值加b
*(p+i)+b a[i]+b 数组的第i个元素的值加b
注:当前p指向第i个元素
*p++ a[i++] 先取得数组的第i个元素的值,i加1//先取得数组的第i个元素的值 指针p再加一,指向a[i+1]
*++p a[++i] 先i加1,取得数组的第i+1个元素的值,//指针p先加一,指向
a[i+1],再取得数组的第i+1个元素的值
*p-- a[i--] 先取得数组的第i个元素的值,i减1//先取得数组的第i个元素的值 指针p再减一,指向a[i-1]
*--p a[--i] 先i减1,取得数组的第i-1个元素的值,//指针p先减一,指向
a[i-1],再取得数组的第i-1个元素的值
,
二维数组:
行指针、列指针
&a[0][0] == a 值一样,含义不一样
&a[0][0]表示数组首地址,列地址
a表示数组首地址,行地址
地址和值:
&a[i][j] &a[i][0]+j a[i]+j *(a+i)+j
a[i][j] *(&a[i][0]+j) *(a[i]+j) *(*(a+i)+j)
2.多级指针(指针的指针)
int **p;
原理见截图
理论上多级指针最多可以到12级别,实际上2级指针用的加多
3.指针数组
int *p[10];
char *a[20];
p是一个指针数组,数组的名字是p,数组有10个元素,每一个元素值都是一个指向整形数据的指针
注意事项: int (*p)[10]的区别,这是一个指针,指针的名字是p,指向一个含有10个元素的整形数组
int a[10];
int (*p)[10];
p=a;
指针数组的引用可以认为是一个指向指针的指针,因为数组名本身代表地址,可以做指针参与运算(不能改变其值)数组元素的值是指针,,因此是二级指针,**p
4.字符串、字符指针数组
C语言本身没有字符串类型,用字符数组来表示字符串(约定最后一个字符是'\0');
char *s="Welcome";
初始化字符串指针时,实际是把内存中字符串的首地址赋值给指针变量,并不是复制字符串给指针,因为指针只占4个字节,用来保存一个地址
字符数组与字符串函数
#include <strings.h>
strcasecmp, strcat, strchr, strcmp, strcpy, strdup, strlen,
strncat, strncmp, strncpy, strncasecmp, strstr, strtok
#include <stdlib.h>
atoi, atol, atoll, atoq - convert a string to an integer
strtod, strtof, strtold - convert ASCII string to floating-point number
字符型指针数组实现程序主菜单!!!!
5.指针函数
指针函数:若一个函数返回的是指针,则称该函数为指针函数。
函数内部的变量都是局部变量,当函数执行完后,会自动释放,在主调函数中不能再访问,访问一段释放的内存,是非法操作,若修改非法内存中的值,可能会出现严重后果,不可预料。
解决方法:
1. 用static将局部变量变为静态局部变量
2. 字符串常量类似于静态变量,在程序结束时,才释放内存。函数中使用 char *str="Welcome"; "Welcome"是一个常量,在内存中有确定的地址,把这个地址返回给主调函数是可以的
3. 在堆空间分配内存,返回分配内存的地址给主调函数。str=(char *)malloc(20);
注意:堆空间的使用一定要注意使用后要释放。 malloc / free 一定是对应的,如果不释放,指针已经指向其他地方,此空间成为内存泄漏空间。
6.函数指针
函数指针是一个专门用于保存函数首地址的指针,实际上就是函数的入口地址。
函数指针指向哪个函数,(*P)()就调用哪个函数。
int (*p)();
实例分析:call.c
输入字符串a+b 调用strsplit函数根据运算符+分割字符串,分割后保存在指针数组char *pstr[]; pstr[0]保存的是第一个操作数a的地址(此时仍当作字符串处理))pstr[1]保存的是第二个操作数b的地址(此时仍当作字符串处理)),使用atoi函数将字符串转换成整形数据,将q指向要调用的函数,调用operate(q,a,b),就完成了相应的运算。
int(*p[10])() 函数指针数组
int a[10];
int *p=a;
*(a+i), a[i], *(p+i), p[i]
字符数组的值:
1.定义的同时初始化
2.输入函数scanf\gets\fgets
3.字符串函数strcpy
char a[]="hello world";
char *p=a;
p="today"; 对 把内存中“today”字符串常量的地址赋值给了p
a="today" 错
结论:指针是地址变量,而数组名是地址常量,数组名在运算中可以作为指针参与,但不允许被赋值,指针可变,允许指向别的位置
指针和数组常见等价操作:
指针操作 数组操作 说明
p &a[0] 数组首地址
*p a[0] 数组的第一个元素
p+i &a[i] 数组的第i个元素的地址
*(p+i) a[i] 数组的第i个元素
*p+b a[0]+b 数组的第一个元素的值加b
*(p+i)+b a[i]+b 数组的第i个元素的值加b
注:当前p指向第i个元素
*p++ a[i++] 先取得数组的第i个元素的值,i加1//先取得数组的第i个元素的值 指针p再加一,指向a[i+1]
*++p a[++i] 先i加1,取得数组的第i+1个元素的值,//指针p先加一,指向
a[i+1],再取得数组的第i+1个元素的值
*p-- a[i--] 先取得数组的第i个元素的值,i减1//先取得数组的第i个元素的值 指针p再减一,指向a[i-1]
*--p a[--i] 先i减1,取得数组的第i-1个元素的值,//指针p先减一,指向
a[i-1],再取得数组的第i-1个元素的值
,
二维数组:
行指针、列指针
&a[0][0] == a 值一样,含义不一样
&a[0][0]表示数组首地址,列地址
a表示数组首地址,行地址
地址和值:
&a[i][j] &a[i][0]+j a[i]+j *(a+i)+j
a[i][j] *(&a[i][0]+j) *(a[i]+j) *(*(a+i)+j)
2.多级指针(指针的指针)
int **p;
原理见截图
理论上多级指针最多可以到12级别,实际上2级指针用的加多
3.指针数组
int *p[10];
char *a[20];
p是一个指针数组,数组的名字是p,数组有10个元素,每一个元素值都是一个指向整形数据的指针
注意事项: int (*p)[10]的区别,这是一个指针,指针的名字是p,指向一个含有10个元素的整形数组
int a[10];
int (*p)[10];
p=a;
指针数组的引用可以认为是一个指向指针的指针,因为数组名本身代表地址,可以做指针参与运算(不能改变其值)数组元素的值是指针,,因此是二级指针,**p
4.字符串、字符指针数组
C语言本身没有字符串类型,用字符数组来表示字符串(约定最后一个字符是'\0');
char *s="Welcome";
初始化字符串指针时,实际是把内存中字符串的首地址赋值给指针变量,并不是复制字符串给指针,因为指针只占4个字节,用来保存一个地址
字符数组与字符串函数
#include <strings.h>
strcasecmp, strcat, strchr, strcmp, strcpy, strdup, strlen,
strncat, strncmp, strncpy, strncasecmp, strstr, strtok
#include <stdlib.h>
atoi, atol, atoll, atoq - convert a string to an integer
strtod, strtof, strtold - convert ASCII string to floating-point number
字符型指针数组实现程序主菜单!!!!
5.指针函数
指针函数:若一个函数返回的是指针,则称该函数为指针函数。
函数内部的变量都是局部变量,当函数执行完后,会自动释放,在主调函数中不能再访问,访问一段释放的内存,是非法操作,若修改非法内存中的值,可能会出现严重后果,不可预料。
解决方法:
1. 用static将局部变量变为静态局部变量
2. 字符串常量类似于静态变量,在程序结束时,才释放内存。函数中使用 char *str="Welcome"; "Welcome"是一个常量,在内存中有确定的地址,把这个地址返回给主调函数是可以的
3. 在堆空间分配内存,返回分配内存的地址给主调函数。str=(char *)malloc(20);
注意:堆空间的使用一定要注意使用后要释放。 malloc / free 一定是对应的,如果不释放,指针已经指向其他地方,此空间成为内存泄漏空间。
6.函数指针
函数指针是一个专门用于保存函数首地址的指针,实际上就是函数的入口地址。
函数指针指向哪个函数,(*P)()就调用哪个函数。
int (*p)();
实例分析:call.c
输入字符串a+b 调用strsplit函数根据运算符+分割字符串,分割后保存在指针数组char *pstr[]; pstr[0]保存的是第一个操作数a的地址(此时仍当作字符串处理))pstr[1]保存的是第二个操作数b的地址(此时仍当作字符串处理)),使用atoi函数将字符串转换成整形数据,将q指向要调用的函数,调用operate(q,a,b),就完成了相应的运算。
int(*p[10])() 函数指针数组