深入探究C语言中的常量指针与野指针概念及其应用

简介: 深入探究C语言中的常量指针与野指针概念及其应用



常量指针(Constant Pointers)

在C语言中,const关键字用于声明常量,而野指针则是一种危险的指针类型。下面将详细解释这两个概念及其应用。

常量指针是指指向常量的指针,它不能用来修改所指向的数据。这有助于保护数据不被意外修改,提高程序的安全性和可维护性。

1. 指向常量的指针

当你想阻止通过指针修改数据时,可以使用指向常量的指针。这种指针的类型是指向常量的指针类型,例如 int * const p;。这意味着你不能通过这个指针来修改它所指向的数据。

2. 常量指针

另一种形式是常量指针,即指针本身的值不能被修改,但可以通过该指针修改其指向的数据。这种指针的类型是指向非常量的常量指针类型,例如 const int *p;。这意味着你不能修改指针 p 所指向的数据的值。

以下是一个使用const声明常量的示例:

#include <stdio.h>
int main() 
{
  const int a = 10;//a具有了常属性(不能被修改了)
  //a本质上还是变量
  //在C++中,const修饰的变量就是常量
  //a = 20; 错误,a修改不了
  //int arr[a]; 错误  a不算常量值
    printf("%d\n", a);
    return 0;
}

在这个示例中,我们声明了一个名为a的整型常量,并将其初始化为10。然后我们使用printf函数输出a的值。注意,我们不能修改a的值,否则编译器会报错。

 

通过修改被const修饰的a的地址,修改a的值

int main()
{
  const int a = 10;
  //a = 20;//error
  int* p = &a;
  *p = 0;
  printf("a = %d\n", a);
  return 0;
//}

 

const位置的不同

//const 修饰指针变量的时候,放在*的右边
//const 限制的是指针变量本身,指针变量不能再指向其他变量了
//但是可以通过指针变量,修改指针变量指向的内容
int main()
{
  int a = 10;
  int b = 20;
  int* const p = &a;
  //一但指向a就不能指向b了
  //p = &b;//error
  *p = 100;//修改*p却可以
  printf("a = %d\n", a);
  return 0;
}
//const 修饰指针变量的时候,放在*的左边
//限制的是指针指向的内容,不能修改指针指向的内容
//但是可以修改指针变量本身的值(修改指针变量的指向)
int main()
{
  int a = 10;
  int b = 20;
  int const* p = &a;
  //一但指向a就不能指向b了
  p = &b;//OK
  //*p = 100;//error
  printf("a = %d\n", a);
  return 0;
}
//int const* const p = &a
//*两边都加上const 就都改不了了
int main()
{
  int a = 10;
  int b = 20;
  int const* const p = &a;
  //一但指向a就不能指向b了
  //p = &b;//error
  //*p = 100;//error
  printf("a = %d\n", a);
  return 0;
}

指针的应用

利用指针打印数组的几种方式

//用指针打印数组
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  int* p = &arr[0];
  int sz = sizeof(arr) / sizeof(arr[0]);
  for (int i = 0; i < sz; i++)
  {
    printf("%d ", *p);
    p++;
  }
  return 0;
}
//另一种方法
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  int* p = &arr[0];
  int sz = sizeof(arr) / sizeof(arr[0]);
  for (int i = 0; i < sz; i++)
  {
    printf("%d ", *(p+i));
    
  }
  return 0;
}
//利用指针的关系运算打印数组
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  int* p = arr;//等于 &arr[0];
  int sz = sizeof(arr) / sizeof(arr[0]);
  while (p < arr + sz)
  {
    printf("%d ", *p);
    p++;
  }
  return 0;
}
//指针-指针的前提是,两个指针指向同一个空间
int main()
{
  int arr[10] = { 0 };
  printf("%zd ", &arr[9] - &arr[0]);//9
  return 0;
}

野指针(Wild Pointers)的产生

野指针通常产生于**未初始化的指针指针越界访问以及指向已释放内存的指针**。具体如下:

1. 未初始化的指针:定义指针变量时,如果没有进行初始化,那么该指针的值是随机的,可能指向任意的内存地址。这种情况下,如果尝试通过这个指针去读取或写入数据,可能会导致程序崩溃或其他不可预期的行为。

2. 指针越界访问:当指针超出了它所指向的数据结构(如数组)的边界时,就会发生越界访问。例如,一个指向大小为10的数组的指针,如果尝试访问数组的第12个元素,就会造成越界。

3. 指向已释放内存的指针:当一块内存被释放后,原有的指针如果继续指向这块内存,而没有置空或者重新赋值,这个指针就变成了所谓的“悬挂指针”或“野指针”。

为了避免野指针的产生和影响,应当总是在声明指针时对其进行初始化,并在释放指针指向的内存后立即将指针置为NULL,同时确保指针在其有效作用域内使用。

代码如下:

//未初始化指针,产生的野指针
int main()
{
  int* p;//p是一个局部变量,不初始化的默认存的是随机值
  *p = 20;
  //报错 printf("%d \n", p);
  return 0;
}
//数组越界,产生的野指针
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  int* p = &arr[0];
  int sz = sizeof(arr) / sizeof(arr[0]);
  for (int i = 0; i <= sz; i++)
  {
    printf("%d ", *p);
    p++;
  }
  return 0;
}
//指针指向的空间释放,产生的野指针
int test()
{
  int a = 10;
  return &a;
}
int main()
{
  int* p = test();
  printf("%d \n", *p);
  return 0;
}

野指针(Wild Pointers)的危害

野指针的危害主要体现在以下几个方面:

1. 触发段错误:当野指针指向一个不可访问的内存地址时,尝试对其进行解引用操作可能会导致段错误,这是因为程序试图访问一个非法的内存区域。

2. 数据损坏:如果野指针指向了一个正在被其他部分的程序使用的内存空间,并且通过这个野指针修改了该内存空间的内容,那么可能会导致数据损坏,甚至程序崩溃。

3. 内存泄漏:在某些情况下,野指针可能导致内存泄漏。例如,如果一个野指针指向了一块已经分配但未被释放的内存,而这块内存又在其他地方被重复分配,就会造成内存泄漏。

4. 调试困难:野指针的存在可能会使得程序的调试变得非常困难,因为它们可能在程序的任何地方引发错误,而且这些错误可能不会立即显现,增加了查找和修复问题的难度。

如何避免

为了避免野指针带来的危害,可以采取以下措施:

1. 初始化指针:在声明指针变量时,应当对其进行初始化,避免其成为一个野指针。

2. 及时置空:当一个指针不再使用时,或者它所指向的内存空间已经被释放时,应该将其置为NULL,以防止其成为野指针。

3. 谨慎解引用:在使用指针前,应当确保它指向的是一个有效的内存地址,避免对无效地址进行解引用操作。

4. 使用智能指针:在一些支持智能指针的编程语言中,可以使用智能指针来自动管理内存,减少野指针的产生。

       总的来说,野指针是C/C++编程中的一个常见问题,它们可能导致程序不稳定、数据损坏和内存泄漏等严重问题。因此,理解野指针的危害并采取适当的预防措施是非常重要的

学习指针的目的是使用指针解决问题,那什么问题,非指针不可呢?

两个数的交换

void swap(int* pa, int* pb)
{
  int tmp = *pa;
  *pa = *pb;
  *pb = tmp;
}
int main()
{
  int a = 10;
  int b = 20;
  printf("交换前:%d %d\n", a, b);
  swap(&a, &b);
  printf("交换前:%d %d\n", a, b);
  return 0;
}

模拟实现库函数strlen

模拟实现库函数strlen
int my_stelen(char* str)
{
  int count = 0;
  while (*str!='\0')
  {
    count++;
    str++;
  }
  return count;
}
int main()
{
  char arr[] = "abcdef";
  int len = my_stelen(arr);
  printf("%d ", len);
  return 0;
}

应用场景

  • 常量指针:当你想保护某些数据不被修改时,可以使用常量指针。例如,在函数参数中传递一个指向常量的指针,这样在函数内部就不能修改这个数据。
  • 野指针:避免使用未初始化的指针和已经释放的内存地址的指针,以减少程序出错的风险。确保在使用指针之前进行适当的初始化,并在不再需要时将其置为NULL。

希望对你有帮助~加油各位!!

目录
相关文章
|
1月前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
94 3
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
1月前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
61 9
|
1月前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
46 7
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
75 5
|
2月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
机器学习/深度学习 算法 数据挖掘
C语言在机器学习中的应用及其重要性。C语言以其高效性、灵活性和可移植性,适合开发高性能的机器学习算法,尤其在底层算法实现、嵌入式系统和高性能计算中表现突出
本文探讨了C语言在机器学习中的应用及其重要性。C语言以其高效性、灵活性和可移植性,适合开发高性能的机器学习算法,尤其在底层算法实现、嵌入式系统和高性能计算中表现突出。文章还介绍了C语言在知名机器学习库中的作用,以及与Python等语言结合使用的案例,展望了其未来发展的挑战与机遇。
53 1
|
2月前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
70 1
|
2月前
|
网络协议 物联网 数据处理
C语言在网络通信程序实现中的应用,介绍了网络通信的基本概念、C语言的特点及其在网络通信中的优势
本文探讨了C语言在网络通信程序实现中的应用,介绍了网络通信的基本概念、C语言的特点及其在网络通信中的优势。文章详细讲解了使用C语言实现网络通信程序的基本步骤,包括TCP和UDP通信程序的实现,并讨论了关键技术、优化方法及未来发展趋势,旨在帮助读者掌握C语言在网络通信中的应用技巧。
54 2
|
2月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
182 13
|
3月前
|
C语言
无头链表二级指针方式实现(C语言描述)
本文介绍了如何在C语言中使用二级指针实现无头链表,并提供了创建节点、插入、删除、查找、销毁链表等操作的函数实现,以及一个示例程序来演示这些操作。
43 0