指针
使用未初始化的指针会带来非常大的风险。0:null/nullptr,nullptr可隐式转为bool
char *p={"abcd"}
建立一个以\0终止的字符串字面值。p为第一个字符的地址。
注意:指向char类型的指针,输出方式以一种特殊的方式来解析这类指针——将p对应位置开始一直到结束都会输出,即p输出字符串本身,而不是字符串的地址,但*p仍是第一个字符
char *p="stra";
其输出结果为:stra,tra,ra。
while(i<3)
{count<<p++<<endl;
i++;}
指针消除了使用数组、数组字符串造成的内存浪费的现象,当两个大对象进行交换时,只需交换地址即可,节省了大量的时间。
指针数组中的元素个数求取:size(pstr)/sizeof(pstr[0])
char (*p)[3]
首先p为一个指针,指向3个元素,每个元素为char型
char *p[3]
首先p为一个数组,有3个元素,每个元素都为指针,指向char型
char p[3]
p为一个数组,数组中有3个元素,每个元素为char型
判断是根据运算符的优先级进行的
运算符优先级
1.::{作用域}
2.(){函数调用,类型构造:type(exp)},[]{下标},.{成员选择},->{成员选择}
3.++{后置},--{后置},typeid{类型id},explicit_cast{四种类型转换}
4.++{前置},--{前置},~{取反},!{逻辑非},-{一元负},+{一元正},\*{指针指向值},&{取地址},(){老式类型转换},sizeof{对象大小}
5.sizeof{类型或参数包的大小},new{分配内存},delete{释放内存},noexcept{能否抛出异常}
6.->\*{指向成员中的指针},.\*{指向成员中的指针}
7.*,/,%
8.+,-
9.<<,>>
10.<,>,<=,>=
11.==,!=
12.&
13.^
14.|
15.&&
16.||
17.?:
18.=
19.*=,/=,%=,+=,-=,<<=,>>=,&=,|=,^={复合赋值}
20.throw{抛出异常}
21.,{逗号}
char (*p)[3]
故p与p+1的差距为sizeof(char p[3])
char p[3]
p与p+1的差距为sizeof(char),而&p与&p+1差距为sizeof(p),其中:
- &p表示整个数组类型
- p代表数组元素类型
例如:
char (*p)[10];
char a[10];
p=&a;
若p=&a改为p=a则会报出如下错误:指针类型不兼容赋值。
- a指向char*,表示第一个字符的地址
- &a指向了char(*)[],表示数组地址
&a和a的右值虽然相等,但类型不匹配,从而导致sizeof(类型)大小不同
n维数组:n维数组都是按一维数组存储方式来存储
二维数组
char[3][2]:看成具有3个元素的一维数组: char (*p)[2]=a可以换成: auto p=a{通过编译器自动推断出p的类型}
- a[0]指向了char*
- a指向了char(*)[2]
- &a指向了char(*)[3][2]
三维数组
char[3][4][2]:看出一维数组
- &a指向了char(*)[3][4][2]
- a指向了第一维数组类型: char(*)[4][2],故a+1表示:首地址+sizeof(char a[4][2])
- a[i]指向了第二维数组类型: char(*)[2]
- a[i][j]指向了第三维数组类型: char*
多级指针
- char *p=”my name”
- char **pp=&p
- char ***ppp=&pp
*p表示m’,*pp表示了p地址,**pp表示了‘m’
*ppp表示了pp的地址,**ppp表示了p的地址,***ppp表示了‘m’
故char*p[]可用char**p表示。
int &*p(此用法非法)
int* &p:首先p是一个引用,引用的类型为整型指针。即给指针取一个别名。
从而上述说明了b[i][j]等价于*(*(b+i)+j)或*(b[i]+j)或*(b+i)[j]
数组引用:与指针用法的区别
int a[10];
int (&b)[10]=a;