目录
上述的代码就是创建了整型变量a,向内存中申请4个字节,⽤于存放整数10,其中每个字节都
1. 内存和地址
1.1内存
我们在学校里住宿的话,学校会给我们分配宿舍,每个人都会有自己的位置。那么当我们需要在学校里面找的时候,这样效率很低,但是我们如果根据楼层和楼层的房间的情况,给每个房间编上号,
1楼,101,102,103,104....
2楼,201,202,203,204....
3楼,301,....
....
....
1b yte = 8b it1 KB = 1024b yte1 MB = 1024 KB1 GB = 1024 MB1 TB = 1024 GB1 PB = 1024 TB
我们在生活中把这些门牌号称之为地址,而在c语言中我们称之为指针。
内存单元的编号 == 地址 == 指针
1.2 究竟该如何理解编址
每条地址总线可以用低或高脉冲(impulse)来表示0或1,我们常说32位和64位的机器,这里是指计算机最多能处理多少数据。例如,当我们使用32位计算机进行操作的时候,其表达的意思就是我们的电脑CPU和内存之间的地址线有32条,这里每一条地址线都可以用高低脉冲来表示0或1,那么我们一共有2^32种组合方式,每一种组合方式都可以代表一个地址。那么64位的机器就有2^64种组合方式,我们就有2^64个地址可以表达
2. 指针变量和地址
2.1 取地址操作符(&)
我们上面介绍了内存,地址和指针之间的关系。
我们知道在c语言中,我们建立一个变量就是在向内存申请一块空间,那么我们怎么能知道这个地址是什么呢?这时我们需要用到一个取地址操作符(&)
我们写下如下一串简单的代码
#include<stdio.h>
int main()
{
int a = 10;
printf("%p ", &a);
return 0;
}
运行结果如下(%p是打印地址)
我们可以看到,他打出了一串数字。那么他是不是真正的地址呢?
我们按下F10,打开调试,在窗口处打开内存
我们在内存中输入&a就可以观察到a的地址了,按F10到第7行时,屏幕上打印出了这串数字
和我们在内存中看到的地址一样,那么我们可以说打印出了这个地址
上述的代码就是创建了整型变量a,向内存中申请4个字节,⽤于存放整数10,其中每个字节都
1.0x000000470B14F5A4
2.0x000000470B14F5A5
3.0x000000470B14F5A6
4.0x000000470B14F5A7
我们可以发现,整形在内存中的存储是占4个字节,且我们取出的是最小的那个地址
2.2 指针变量和解引用操作符(*)
2.2.1指针变量
上面我们一直在说地址怎么怎么样,那么当我们需要地址(指针)时,我们该怎么得到呢?这些地址(指针),又存放在哪里呢?
用来存放指针的变量我们称之为指针变量。(指针变量就像,我们要通过门牌号找到一个人的时候,得先从存这些门牌号的储存空间中找到那个门牌号,要通过门牌号找到数据,指针变量就是用来存放这些门牌号的)
那我们通过取地址操作符(&)拿到的地址是⼀个数值,⽐如:0x006FFD70,这个数值有时候也是需要存储起来,⽅便后期再使⽤的,那我们把这样的地址值存放在哪⾥呢?
答案是:指针变量中。
#include <stdio.h>
int main()
{
int a = 10;
int* pa = &a;//取出a的地址并存储到指针变量pa中
return 0;
}
2.2.2 如何拆解指针类型
int main()
{
int a = 10;
int* pa = &a;
return 0;
}
pa的类型是int* ,这里(*)是告诉我们这是一个指针,int是说明这是int(哪种)变量类型的地址。
如果是char类型,我们就写作
char* pa = &a
如果是float(浮点型),我们就写作
float* pa = &a
.....等等
2.2.3 解引用操作符(*)
#include <stdio.h>
int main()
{
int a = 100;
int* pa = &a;
*pa = 0;
return 0;
}
上⾯代码中就使⽤了解引⽤操作符, *pa 的意思就是通过pa中存放的地址,找到指向的空间,*pa其实就是a变量了;所以*pa = 0,这个操作符是把a改成了0.有同学肯定在想,这⾥如果⽬的就是把a改成0的话,写成 a = 0; 不就完了,为啥⾮要使⽤指针呢?其实这⾥是把a的修改交给了pa来操作,这样对a的修改,就多了⼀种的途径,写代码就会更加灵活
2.3 指针变量的大小
我们上面说到,32位机器下,共有32根线来表示0或1,他们组成的地址一共有2^32个地址(每一个地址都会用32根线来组成,我们说过这些线都表示0或1的二进制数,那么每一个地址就是32个bit位,也就是4个字节,那么我们可以说每个指针变量都是4个字节。那么到底是不是呢?下面可以用sizeof()操作符来计算一下32位下各种类型指针变量的大小(单位字节))
我们发现确实是这样的
那么进而可以推导出:64位机器下的指针变量大小是64个bit位也就是8个字节
结论:
• 32位平台下地址是32个bit位,指针变量⼤⼩是4个字节• 64位平台下地址是64个bit位,指针变量⼤⼩是8个字节• 指针变量的⼤⼩和类型是⽆关的,只要指针类型的变量,在相同的平台下,⼤⼩都是相同的。
3. 指针变量类型的意义
3.1 指针的解引用
我们先来看以下代码在内存中的情况,
int main()
{
int n = 0x11223344;
int* pi = &n;
*pi = 0;
return 0;
}
我们看到我们存入的16进制数据0x11223344已经存入n这时我们取地址&n,存入指针变量pi中,
再解引用其地址,使其为0
我们发现,其都是为0了。
下面我们来看另一个有一点不一样的代码
int main()
{
int n = 0x11223344;
char* pc = (char*)&n;
*pc = 0;
return 0;
}
这个代码不同的地方在于把n的地址强制从int转换为char,再存入char类型的指针变量pc中
我们惊奇地发现只有第一个数据发生了变化,char类型是一个字节的大小,我们就可以总结结论
结论:指针的类型决定了,对指针解引⽤的时候有多⼤的权限(⼀次能操作⼏个字节)。⽐如: char* 的指针解引⽤就只能访问⼀个字节,⽽ int* 的指针的解引⽤就能访问四个字节
3.2 验证
int main()
{
int n = 10;
int* pi = &n;//直接取n的地址到pi中
char* pc = (char*)&n;强制转换n指针类型成char类型,并将其存入char类型指针pc中
printf("%p\n", &n);//取n的地址
printf("%p\n", pc);//打印pc的地址
printf("%p\n", pc + 1);//打印pc+1的地址
printf("%p\n", pi);//打印pi的地址
printf("%p\n", pi + 1);//打印pi+1的地址
return 0;
}
我们发现作为char类型的地址pc加一的时候只跳过了1个字节,而作为int类型的地址pi跳过了4个字节
这就是指针变量的类型差异带来的变化。结论:指针的类型决定了指针向前或者向后⾛⼀步有多⼤(距离)
4. const修饰指针
4.1 const修饰变量
#include <stdio.h>
int main()
{
int m = 0;
m = 20;//m是可以修改的
const int n = 0;
n = 20;//n是不能被修改的
return 0;
}
上述代码中n是不能被修改的,其实n本质是变量,只不过被const修饰后,在语法上加了限制,只要我们在代码中对n就⾏修改,就不符合语法规则,就报错,致使没法直接修改n。
但是如果我们绕过n,使⽤n的地址,去修改n就能做到了,虽然这样做是在打破语法规则。
#include <stdio.h>
int main()
{
const int n = 0;
printf("n = %d\n", n);
int*p = &n;
*p = 20;
printf("n = %d\n", n);
return 0;
}
我们发现确实可以修改
4.2 const修饰指针变量
void test1()
{
int n = 10;
int m = 20;
int* p = &n;
*p = 20;//ok?
p = &m; //ok?
}
void test2()
{
//代码2
int n = 10;
int m = 20;
const int* p = &n;
*p = 20;//ok?
p = &m; //ok?
}
void test3()
{
int n = 10;
int m = 20;
int* const p = &n;
*p = 20; //ok?
p = &m; //ok?
}
void test4()
{
int n = 10;
int m = 20;
int const* const p = &n;
*p = 20; //ok?
p = &m; //ok?
}
int main()
{
//测试⽆const修饰的情况
test1();
//测试const放在*的左边情况
test2();
//测试const放在*的右边情况
test3();
//测试*的左右两边都有const
test4();
return 0;
}
结论:const修饰指针变量的时候• const如果放在*的左边,修饰的是指针指向的内容,保证指针指向的内容不能通过指针来改变。但是指针变量本⾝的内容可变。• const如果放在*的右边,修饰的是指针变量本⾝,保证了指针变量的内容不能修改,但是指针指向的内容,可以通过指针改变。
5. 指针运算
5.1 指针+- 整数
因为数组在内存中是连续存放的,只要知道第⼀个元素的地址,顺藤摸⽠就能找到后⾯的所有元素。
#include <stdio.h>
//指针+- 整数
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int *p = &arr[0];
int i = 0;
int sz = sizeof(arr)/sizeof(arr[0]);
for(i=0; i<sz; i++)
{
printf("%d ", *(p+i));//p+i 这⾥就是指针+整数
}
return 0;
}
5.2 指针-指针
strlen的一种实现
//指针-指针
#include <stdio.h>
int my_strlen(char *s)
{
char *p = s;
while(*p != '\0' )
p++;
return p-s;//这里用后面的地址减去前面的地址,就可以得到中间地址个数,char类型大小是一个字节
}
int main()
{
printf("%d\n", my_strlen("abc"));
return 0;
}
5.3 指针的关系运算
#include <stdio.h>
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int *p = &arr[0];
int i = 0;
int sz = sizeof(arr)/sizeof(arr[0]);
while(p<arr+sz) //指针的⼤⼩⽐较
{
printf("%d ", *p);
p++;
}
return 0;
}
6. 野指针
概念: 野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)
6.1 野指针的成因
6.1.1.指针未初始化
int main()
{
int a;
int* p = &a;
return 0;
}
这里指针变量p存的a的指针就是随机值,因为a未进行初始化 。所以他是野指针
如果不知道指针该初始化什么值,为了安全初始化为NULL(空指针)
int* p = NULL;
6.1.2.指针越界访问
int main()
{
int arr[10] = {0};
int *p = &arr[0];
int i = 0;
for(i=0; i<=11; i++)
{
//当指针指向的范围超出数组arr的范围时,p就是野指针
*(p++) = i;
}
return 0;
}
6.1.3.指针指向的空间释放(避免返回局部变量的地址)
当我们在一个函数中建立了一个指针,当我们走出这个函数后,这块空间的内容会销毁,这时我们就不能使用原来的指针了。
#include <stdio.h>
int* test()
{
int n = 100;
return &n;
}
int main()
{
int*p = test();
printf("%d\n", *p);
return 0;
}
这就是典型的野指针
6.2 怎么避免野指针
1.我们可以及时使指针为NULL
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int *p = &arr[0];
for(i=0; i<10; i++)
{
*(p++) = i;
}
//此时p已经越界了,可以把p置为NULL
p = NULL;
//下次使⽤的时候,判断p不为NULL的时候再使⽤
//...
p = &arr[0];//重新让p获得地址
if(p != NULL) //判断
{
//...
}
return 0;
}
当指针变量指向⼀块区域的时候,我们可以通过指针访问该区域,后期不再使⽤这个指针访问空间的时候,我们可以把该指针置为NULL。因为约定俗成的⼀个规则就是:只要是NULL指针就不去访问,同时使⽤指针之前可以判断指针是否为NULL。
2. ⼩⼼指针越界
#include <stdio.h>
int main()
{
int num = 10;
int*p1 = #
int*p2 = NULL;
return 0;
}
8. 指针的使用和传址调用
void swap(int x, int y)
{
int temp = 0;
temp = x;
x = y;
y = temp;
}
int main()
{
int x = 10;
int y = 20;
swap(x, y);
printf("交换后x的值:%d\n", x);
printf("交换后y的值:%d\n", y);
return 0;
}
void swap(int* x, int* y)
{
int temp = *x;
*x = *y;
*y = temp;
}
int main()
{
int x = 10;
int y = 20;
swap(&x, &y);
printf("交换后x的值:%d\n", x);
printf("交换后y的值:%d\n", y);
return 0;
}