【零基础学习C语言】深入指针(2)

零基础学C语言:深入指针知识详解

目录

1.const修饰指针

2.野指针

3.assert断言

4.指针的使用和传址调用

一 const

1.const修饰变量

在C语言中,const修饰完是常变量,在C++中就是常量。

出现报错

2.const修饰指针

const放在 左边

限制的是指针变量指向的内容,不能通过指针变量p改变*p的内容,但p的内容是可以改变的,p可以指向其他变量。

const放在 右边

限制的是p这个指针变量,p的内容不能被修改,p不能指向其他变量,但是可以改变*p的内容。

二 野指针

1.概念

野指针就是指针指向的位置是不可知的

2.成因

(1)指针未初始化
#include<stdio.h>
int main()
{
   int *p; //p没有初始化,那么p的值就是随机值
   *p = 10;
   return 0;
}
(2)指针越界
int main()
{
  int arr[10] = {0};
  int *p = arr;
  for(int i = 0;i<11;i++)
  {
   //指针范围超出数组arr的范围
    *(p++) = 1;
  }
  return 0;
}
(3)指针指向的空间释放
int* test()
{
  int n = 100; 
  printf("%d",n);
  return &n;
}
int main()
{
  int *p = test();//p存放的地址指向的内存空间已经释放了
  printf("%d",*p);
  return 0;
}

3.规避野指针

(1)指针初始化

如果知道指针应该指向哪里,就给指针一个值

int n = 10;
int *p = &n;

如果不知道指向哪里时,就指向NULL

int main()
  int *p = NULL;
  if(p!=NULL)
  {
    …………
  }
  return 0;
}

(2)小心指针越界

(3)指针变量不再使用时,及时置NULL,指针使用之前检查有效性

当指针变量指向一块区域的时候,我们可以通过指针访问该区域,后期不再使用这个指针访问空间的时候,我们可以把该指针置为NULL。同时使用指针之前可以判断指针是否为NULL。

int main()
{
  int arr[] = {1,2,3,4,5,6,7,8,9,10};
  int * p = &arr[0];
  int i = 0;
  for(i = 0;i<10:i++)
  {
     *(p++) = i;
  }
  //此时p已经越界,可以把p指向NULL
  p = NULL;
  p = &arr[0];//重新让P获得地址
  if(p!=NULL)//判断
  { 
     //…………
  }
return 0;
}

(4)避免返回局部变量的地址

三 assert断言

assert.h头文件定义了宏 assert( ),用于在运行时确保程序符合指定条件,如果不符合,就报错终止运行。这个宏常常被称为“断言”。

assert(p!=NULL);

上面代码在程序运行到这一行语句时,验证变量 p是否等于 NULL。如果确实不等于 NULL,程序继续运行,否则就会终止运行,并且给出报错信息提示。

使用 assert()的好处:它不仅能自动标识文件和出问题的行号,还有一种无需更改代码就能开启或关闭 assert()的机制。如果已经确认程序没有问题,不需要再做断言,就在#include<assert.h>语句的前面,定义一个宏NDEBUG

#define NDEBUG
#include <assert.h>

然后,重新编译程序,编译器就会禁用文件中所有的 assert()语句。如果程序又出现问题,可以移除这条#define NDEBUG 指令(或者把它注释掉),再次编译,这样就重新启用了 assert()语句。

一般我们可以在 Debug 中使用,在 Release 版本中选择禁用 assert 就行,在 vs 这样的集成开发环境中,在Release 版本中,直接就是优化掉了。这样在debug版本写有利于程序员排査问题,在 Release 版本不影响用户使用时程序的效率。

四 指针的使用和传址调用

1.strlen的模拟实现

strlen,计算一个字符串的长度,原理是逐步计算字符遇到“\0”时停止。

2.写一个函数,将两个整形变量的值进行交换

在以往的学习中知道,利用新变量,很简单就可以实现数值交换。

int x = 1;
int y = 2;
int z;

z = x;
x = y;
y = z;

如果把此串代码放到一个函数中会成功吗?(传值调用)

发现并没有实现交换

原因:实参(a,b)传递给形参(x,y)的时候,形参是实参的一份临时拷贝,形参有自己独立的空间,对形参的修改是不会影响实参的。

改进:把a,b的地址传递到函数内(传址调用)

可以让函数与主函数之间建立联系,在函数内部可以修改主调函数中的变量


深入指针(3)正在加速编写中……

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值