参考: 里科《C和指针》
指针存储的是一个地址,实际就是一个值。
如果像下面一样对未初始化的指针进行赋值,如果a的初始值是非法地址,那么会报错。UNIX会提示段错误segmentation violation,或内存错误memory fault,指程序试图访问一个未分配给程序的内存位置;如果windows的话一般是保护性异常General Protection Exception。如果a的初始值是合法地址,那可能更麻烦了,因此在对指针进行间接访问之前,必须确保已经初始化(或者是NULL)。
c
int *a;
...
*a = 12;
向函数传递指针前也要先检查是否为NULL。因为按标准,在函数里不检查,检查交给用户(创建时检查)。
指针的指针
c
int a = 12;
int *b = &a;
int **c = &b;
c
char ch = 'a';
char *cp = &ch;
表达式 | 值 | 说明 | 重要性 |
---|---|---|---|
*cp + 1 | 'b' | *的优先级高于+ | |
*(cp + 1) | 这是非法的,ch后面的值未知 | ||
++cp | 非法,是找ch后面的值 | ||
*++cp | 右值是ch后面的值,左值是ch后面那个位置 | ||
*cp++ | 右值是ch的值,左值是ch的地址。但是cp移到下一个位置了。后缀++的优先级高于*,这里是1)++产生了cp的拷贝;2)++操作cp;3)在cp的拷贝上实现间接访问 | ⭐ | |
++*cp | 'b' | ++和*都是从右向左结合的,所以先间接访问。不能作为左值 | |
(*cp)++ | 'a' | 因为先间接访问,所以计算的值就是'a',随后ch的值+1变'b' | |
++*++cp | 先后移指针,然后取值,再+1 | 很少见 | |
++*cp++ | 因为后缀++的优先级高,所以先移动cp,然后间接访问到ch,+1后ch='b',此时cp还是指向ch后一个位置 | 很少见 |
实例
c
size_t
strlen(char* string) {
int length = 0;
/*
* 如果string=null会报错,因为++自动移动,但是访问冲突
* 这里不检查是否为null,是希望用户检查
*/
while ( *string++ != '\0')
{
length += 1;
}
return length;
}
指针算术加减
指针+1是移动到下一个位置,而非下一个字节。比如int要4个字节,那么int *ptr, ptr+1时实际移动了4个字节,并不是在指针内部移动。
指针减指针
只有在数组内有意义,得到的结果是ptrdiff_t类型,是一种有符号整数类型。这个结果除以数组类型的长度(比如float是4)就是坐标差。如果两个指针指向的元素不在同一数组,这个结果就是未定义的。大部分编译器也不会检查是不是在一个数组。
指针的关系运算
同样在数组中有意义,指< ≤ > ≥。虽然标准允许指向数组元素的指针,与数组最后一个元素后面的内存位置的指针比较,但是不允许它与指向index=0前一个位置的指针比较。不能保证移植性。