段错误和野指针,关键词及内存管理

野指针和段错误;
什么叫野指针:
p保存的是int a的地址,那么就说 p指向a;

1、当指针指向不明确的时候;
int *p; //此时不知道p保存的是什么地址!
*p = 10;
2、指针保存的地址不存在!
int *p = 0x12345678;

2、指针指向明确,地址也存在,但是我们指向的空间没有操作权限;
int *p = NULL; //指向了0地址;但是0地址受系统保护,没有操作权限!
int b;
p = &b;

*p = 10;

满足上面三个条件之一,我们就称这个指针为野指针!

野指针会带来什么危害:
1、段错误!
其实就是地址错误:要么是地址不存在,要么是地址存在,而我们没有操作权限!
2、暂时没有危害,程序正常运行!
但是会在程序里面隐藏bug,以后在某个时间点会爆发!
3、程序直接崩溃!

怎么避免野指针?
归根到底就是养成良好的编码习惯;
1、定义指针的时候,随后给指针初始化为NULL;
2、当你使用指针要判断一下指针是否为NULL;
3、使用完指针的时候,要随后把指针变量指向NULL;

接下来几个关键字:
1、static:静态变量;
作用:
1、修饰局部变量的时候,可以延长局部变量的生命周期!
2、修饰全局变量或者函数的时候,可以起到“隐藏”的功能,进而达到只能够在本文件
中使用的目的,不能在别的文件中使用;


当定义一个静态变量,而没有初始化,系统会自动初始化为0;
static int num;
printf("num = %d\n",num);


2、const:修饰变量称为“常变量”;
int sum = 10;
int len = 9;

int const *p  = ∑

sum  = 18;
p = len;


3、register: 寄存器变量
修饰局部变量,不能够修饰全局变量和函数;
register int num = 10;
请求编译器,将register修饰的变量保存到CPU内部寄存器上!而不是保存到内存中!
进而提高读取效率!
CPU:皇帝;
内存:大臣;
寄存器:小太监;
既然是请求,编译器可以拒绝!因为CPU内部的寄存器数量有限!
既然register修饰的变量保存到CPU内部寄存器上!【所以变量的类型
CPU要能够支持!】而不是保存到内存中!【所以不能够用取地址符(&)来变量的地址】


4、extern;
5、volatile:防止编译器优化处理!


内存管理:
1、栈空间(stack):保存的是函数的形参,局部变量和自动变量;
特点:
1、自动管理:有系统自动给函数形参,局部变量和自动变量分配空间;
执行结束之后,系统自动释放,不用程序员自己去管;
2、反复使用:栈空间比较小;
3、栈空间是脏的;int  num;printf("num");
4、临时性;

2、堆空间(heap):用malloc 、realloc、calloc来申请空间;
特点:
1、灵活;
2、内存空间比较大;
3、需要手动申请和释放;
4、堆空间是脏的;
5、临时性;

怎么用:
//第一步:申请;如果申请成功返回的是申请到的堆空间的首地址;
char *str = (char *)malloc(100);
//第二步:检查是否申请成功;
if( str == NULL )
{
失败;
return;
}
//第三步:使用;
*(str + 1) = 'h';
str++;
//第四步:使用完释放;
free(str); //里面的参数是申请到首地址;

如果不手动释放,在程序结束之前这块空间就【内存泄露】;
当程序结束之后,由系统回收!

3、数据段:保存的是全局变量和被初始化为非0的静态变量;
4、.bss段:保存的是初始化为0的静态变量或者没有初始化的静态变量;
5、代码段(.text):函数的指令,以及字符串常量!

你可能感兴趣的:(段错误和野指针,关键词及内存管理)