这不是我第一次写关于C指针的文章了,只是由于指针对于C来讲过重要,并且随着本身编程经历越多,对指针的理解越多,所以有了本文。然而,想要全面理解指针,除了要对C语言有熟练的掌握外,还要有计算机硬件以及操做系统等方方面面的基本知识。因此我想经过一篇文章来尽量的讲解指针,以对得起这个文章的标题吧。程序员
指针解决了一些编程中基本的问题。编程
而做为一个程序员,咱们不须要了解内存的物理结构,操做系统将DRAM等硬件和软件结合起来,给程序员提供的一种对物理内存使用的抽象。这种抽象机制使得程序使用的是虚拟存储器,而不是直接操做物理存储器。全部的虚拟地址造成的集合就是虚拟地址空间。 数组
在程序员眼中的内存应该是下面这样的。(假设使用的是32位系统平台,虚拟存储空间为4GB)数据结构
也就是说,虚拟存储器是一个很大的,线性的字节数组(平坦寻址)。每个字节都是固定的大小,由8个二进制位组成。最关键的是,每个字节都有一个惟一的编号,编号从0开始,一直到最后一个字节。如上图中,这是一个4GB的虚拟存储器的模型,它一共有4x1024x1024x1024 个字节,那么它的虚拟地址范围就是 0 ~ 4x1024x1024x1024-1 。编程语言
因为内存中的每个字节都有一个惟一的编号,所以,在程序中使用的变量,常量,甚至数函数等数据,当他们被载入到内存中后,都有本身惟一的一个编号,这个编号就是这个数据的地址。指针就是这样造成的。函数
下面用代码说明性能
#include <stdio.h> int main(void) { char ch = 'a'; int num = 97; printf("ch 的地址:%p\n",&ch); //ch 的地址:0028FF47 printf("num的地址:%p\n",&num); //num的地址:0028FF40 return 0; }
为了简单起见,这里就用上面例子中的 int num = 97 这个局部变量来分析变量在内存中的存储模型。测试
用来保存 指针 的变量,就是指针变量。若是指针变量p1保存了变量 num的地址,则就说:p1指向了变量num,也能够说p1指向了num所在的内存块 ,这种指向关系,在图中通常用 箭头表示。spa
上图中,指针变量p1指向了num所在的内存块 ,即从地址0028FF40开始的4个byte 的内存块。操作系统
int a ; //int类型变量 a int *a ; //int* 变量a int arr[3]; //arr是包含3个int元素的数组 int (* arr )[3]; //arr是一个指向包含3个int元素的数组的指针变量 //-----------------各类类型的指针------------------------------ int* p_int; //指向int类型变量的指针 double* p_double; //指向idouble类型变量的指针 struct Student *p_struct; //结构体类型的指针 int(*p_func)(int,int); //指向返回类型为int,有2个int形参的函数的指针 int(*p_arr)[3]; //指向含有3个int元素的数组的指针 int** p_pointer; //指向 一个整形变量指针的指针
int add(int a , int b) { return a + b; } int main(void) { int num = 97; float score = 10.00F; int arr[3] = {1,2,3}; //----------------------- int* p_num = # float* p_score = &score; int (*p_arr)[3] = &arr; int (*fp_add)(int ,int ) = add; //p_add是指向函数add的函数指针 return 0; }
int add(int a , int b){ return a + b; } int main(void) { int arr[3] = {1,2,3}; //----------------------- int* p_first = arr; int (*fp_add)(int ,int ) = add; const char* msg = "Hello world"; return 0; }
int main(void) { int age = 19; int*p_age = &age; *p_age = 20; //经过指针修改指向的内存数据 printf("age = %d\n",*p_age); //经过指针读取指向的内存数据 printf("age = %d\n",age); return 0; }
int* p1 = & num; int* p3 = p1; //经过指针 p1 、 p3 均可以对内存数据 num 进行读写,若是2个函数分别使用了p1 和p3,那么这2个函数就共享了数据num。
#ifdef __cplusplus #define NULL 0 #else #define NULL ((void *)0) #endif
void opp() { int*p = NULL; *p = 10; //Oops! 不能对NULL解地址 } void foo() { int*p; *p = 10; //Oops! 不能对一个未知的地址解地址 } void bar() { int*p = (int*)1000; *p =10; //Oops! 不能对一个可能不属于本程序的内存的地址的指针解地址 }
指针也是一种数据,指针变量也是一种变量,所以指针 这种数据也符合前面 变量和内存 主题中的特性。 这里我只想强调2个属性: 指针的类型,指针的值。
int main(void) { int num = 97; int *p1 = # char* p2 = (char*)(&num); printf("%d\n",*p1); //输出 97 putchar(*p2); //输出 a return 0; }
typedef struct { char name[31]; int age; float score; }Student; int main(void) { Student stu = {"Bob" , 19, 98.0}; Student*ps = &stu; ps->age = 20; ps->score = 99.0; printf("name:%s age:%d\n",ps->name,ps->age); return 0; }
int main(void) { int arr[3] = {1,2,3}; int*p_first = arr; printf("%d\n",*p_first); //1 return 0; }
int main(void) { int arr[3] = {1,2,3}; int*p = arr; for(;p!=arr+3;p++){ printf("%d\n",*p); } return 0; }
int main(void) { int arr[3] = {1,2,3}; int*p = arr; printf("sizeof(arr)=%d\n",sizeof(arr)); //sizeof(arr)=12 printf("sizeof(p)=%d\n",sizeof(p)); //sizeof(p)=4 return 0; }
void change(int a) { a++; //在函数中改变的只是这个函数的局部变量a,而随着函数执行结束,a被销毁。age仍是原来的age,纹丝不动。 } int main(void) { int age = 19; change(age); printf("age = %d\n",age); // age = 19 return 0; }
void change(int* pa) { (*pa)++; //由于传递的是age的地址,所以pa指向内存数据age。当在函数中对指针pa解地址时, //会直接去内存中找到age这个数据,而后把它增1。 } int main(void) { int age = 19; change(&age); printf("age = %d\n",age); // age = 20 return 0; }
再来一个老生常谈的,用函数交换2个变量的值的例子:
#include<stdio.h> void swap_bad(int a,int b); void swap_ok(int*pa,int*pb); int main() { int a = 5; int b = 3; swap_bad(a,b); //Can`t swap; swap_ok(&a,&b); //OK return 0; } //错误的写法 void swap_bad(int a,int b) { int t; t=a; a=b; b=t; } //正确的写法:经过指针 void swap_ok(int*pa,int*pb) { int t; t=*pa; *pa=*pb; *pb=t; }
typedef struct { char name[31]; int age; float score; }Student; //打印Student变量信息 void show(const Student * ps) { printf("name:%s , age:%d , score:%.2f\n",ps->name,ps->age,ps->score); }
void echo(const char *msg) { printf("%s",msg); } int main(void) { void(*p)(const char*) = echo; //函数指针变量指向echo这个函数 p("Hello "); //经过函数的指针p调用函数,等价于echo("Hello ") echo("World\n"); return 0; }
若是const 后面是一个类型,则跳过最近的原子类型,修饰后面的数据。(原子类型是不可再分割的类型,如int, short , char,以及typedef包装后的类型)
int main() { int a = 1; int const *p1 = &a; //const后面是*p1,实质是数据a,则修饰*p1,经过p1不能修改a的值 const int*p2 = &a; //const后面是int类型,则跳过int ,修饰*p2, 效果同上 int* const p3 = NULL; //const后面是数据p3。也就是指针p3自己是const . const int* const p4 = &a; // 经过p4不能改变a 的值,同时p4自己也是 const int const* const p5 = &a; //效果同上 return 0; }
typedef int* pint_t; //将 int* 类型 包装为 pint_t,则pint_t 如今是一个完整的原子类型 int main() { int a = 1; const pint_t p1 = &a; //一样,const跳过类型pint_t,修饰p1,指针p1自己是const pint_t const p2 = &a; //const 直接修饰p,同上 return 0; }
若是2个程序单元(例如2个函数)是经过拷贝 他们所共享的数据的 指针来工做的,这就是浅拷贝,由于真正要访问的数据并无被拷贝。若是被访问的数据被拷贝了,在每一个单元中都有本身的一份,对目标数据的操做相互 不受影响,则叫作深拷贝。
指针和引用这个2个名词的区别。他们本质上来讲是一样的东西。指针经常使用在C语言中,而引用,则用于诸如Java,C#等 在语言层面封装了对指针的直接操做的编程语言中。引用是编程语言提供给程序员的抽象机制,而指针是操做系统提供给软件开发模型的抽象机制。
//测试机器使用的是否为小端模式。是,则返回true,不然返回false
//这个方法判别的依据就是:C语言中一个对象的地址就是这个对象占用的字节中,地址值最小的那个字节的地址。
bool isSmallIndain() { unsigned int val = 'A'; unsigned char* p = (unsigned char*)&val; //C/C++:对于多字节数据,取地址是取的数据对象的第一个字节的地址,也就是数据的低地址 return *p == 'A'; }