指针的一些细节补充———C语言

 野指针:

 1.未初始化的指针

eg:
int *p;       // 未初始化的指针
*p = 5;       // 未定义行为,p 是野指针
———————————————————————————————————————————————————————————————————————————————————————————
//解决方法:
//在动态内存开辟完成后应该进行检查
int *p = (int *)malloc(sizeof(int));
if (p == NULL) {
    // 处理内存分配失败的情况
    return -1;
}
———————————————————————————————————————————————————————————————————————————————————————————
//进行及时的初始化
int value = 42;
int *p = &value;  // 在声明时初始化

2.指向已经释放的内存

//动态内存释放后继续使用指针:
#include <stdio.h>
#include <stdlib.h>

int main() {
    int *p = (int *)malloc(sizeof(int));  // 动态分配内存
    *p = 42;                              // 使用内存

    free(p);                              // 释放内存
    *p = 24;  // 错误:悬空指针访问,未定义行为
//解决方法:
//应该将指针进行滞空
    free(p);
    p = NULL;

    printf("%d\n", *p);  // 未定义行为,输出可能随机
    return 0;
}
—————————————————————————————————————————————————————————————————————————————————————————
//进行重复的释放:
#include <stdlib.h>

int main() {
    int *p = (int *)malloc(sizeof(int));

    free(p);   // 第一次释放
    free(p);   // 错误:重复释放,未定义行为
//解决方法:
//进行重复释放的判断
if (p != NULL) {
    free(p);
    p = NULL;
}
    return 0;
}
—————————————————————————————————————————————————————————————————————————————————————————
//指针副本仍指向已释放的内存
#include <stdio.h>
#include <stdlib.h>

int main() {
    int *p1 = (int *)malloc(sizeof(int));  // 动态分配内存
    int *p2 = p1;                          // p2 也是悬空指针
    free(p1);  // 释放内存,p1和p2现在都指向已释放的内存
    // 任何对p2的访问都是未定义行为
    *p2 = 10;  // 未定义行为
    printf("%d\n", *p2);  // 未定义行为
    return 0;
}

3.指向超出变量生命周期的内存

//指向函数内局部变量的指针:
#include <stdio.h>

int* getPointer() {
    int x = 10;  // 局部变量 x 的生命周期在函数结束时终止
    //在int前加static使用静态变量
    return &x;   // 返回局部变量的地址,危险操作,返回后就已经是悬空指针了
}

int main() {
    int *p = getPointer();  // p 现在是一个悬空指针
    printf("%d\n", *p);     // 未定义行为,可能会打印垃圾值或导致程序崩溃
    return 0;
}
—————————————————————————————————————————————————————————————————————————————————————————
//指向作用域内局部变量的指针
#include <stdio.h>

int* pointerOutsideScope() {
    int *p;
    {
        int y = 20;  // y 是代码块内的局部变量
        p = &y;      // p 指向 y 的内存地址
    }
    // 此时 y 的生命周期结束,p 是一个悬空指针
    return p;
//使用
    int *p = (int *)malloc(sizeof(int));
    *p = 10;  // 动态内存的生命周期由程序员控制
    return p;
}

int main() {
    int *ptr = pointerOutsideScope();
    printf("%d\n", *ptr);  // 未定义行为,可能打印垃圾值
    return 0;
}
 

4.指向已经重新分配的内存

#include <stdio.h>
#include <stdlib.h>

int main() {
    int *p = (int *)malloc(10 * sizeof(int));  // 分配内存
    if (p == NULL) {
        printf("Memory allocation failed\n");
        return -1;
    }
    // 初始化数组
    for (int i = 0; i < 10; i++) {
        p[i] = i + 1;  
    }

    int *q = p;  // q 和 p 指向同一块内存

    p = (int *)realloc(p, 20 * sizeof(int));  // 重新分配内存

    if (p == NULL) {
        printf("Memory reallocation failed\n");
        free(q);  // 如果 realloc 失败,释放原内存
        return -1;
    }

    // q 现在是悬空指针,因为内存可能已经被移动
    printf("%d\n", q[0]);  // 未定义行为,q 指向无效内存
    free(p);  // 释放新的内存块
    return 0;
}

指针运算:

1.指针+-整数

如何获取数组中最后一个元素的地址arr+size-1 或 &arr[size-1]

#define N_VALUES 5 // 定义数组的长度
#include <stdio.h>
int main() {
float values[N_VALUES] = { -1 };
float* vp;
for (vp = &values[0]; vp < &values[N_VALUES];) {
*vp++ = 0;
}
return 0;
}

2.指针-指针

会得到两个指针中间元素的个数

int main() {
int arr[5] = { 13,34,21,38,49 };
int res = &arr[4] - &arr[0];
printf("%d\n", res);
return 0;
}

75bbca418f8346eca9516f75d3b9e061.png

指针的关系运算

#define N_VALUES 5 // 定义数组的长度
#include <stdio.h>
int main() {
    float values[N_VALUES] = { -1 };
    float* vp;
    for (vp = &values[N_VALUES]; vp > &values[0];) {
        *vp++ = 0;
    }
    return 0;
}

允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与指向第一个元素之前的那个内存位置的指针进行比较。

指针函数:

本质是函数不过返回值是地址

#include <stdio.h>
float* designArray(int* p_length){
    static float arr[]={68.5,75.4,69.8,90,75.6,78};
    int size = sizeof(arr)/sizeof(arr[0]);
    *p_length = size;
    return arr;//返回的是一个地址
}
void showArr(float* arr,int length){
    for(int i=0;i<length;i++){
        printf("%-8.2f",*(arr+i));
    }
    printf("\n");
}
int main()
{
    int length = 0;
    float* result = designArray(&length);
//ouput arr
    showArr(result,length);
    return 0;
}

函数指针:

指向函数的指针

#include <stdio.h>
int getMax(int a,int b){
    //对应的指针类型int (*)(int,int)
    return a>b?a:b;
}
int main()
{
    int a=0,b=0;
    printf("input two data:");
    scanf("%d%d",&a,&b);
    int max = getMax(a,b);
    printf("The maximum values of %d and %d are %d\n",a,b,max);
return 0;
}

回调函数

#include <stdio.h>
int add(int a,int b){
return a + b;
}
int sub(int a,int b){
return a - b;
}
int calc(int a,int b,int (*pfun)(int,int)){
    return pfun(a,b);
}
int main(){
    int a=0,b=0;
    printf("input two data:");
    scanf("%d%d",&a,&b);
    int result = calc(a,b,add);//在一个函数中调用另一个函数
    printf("a + b = %d\n",result);
    result = calc(a,b,sub);
    printf("a - b = %d\n",result);
return 0;
}

36302c9e504f436c9b27dfd6482546bf.png

  • 这是本人的学习笔记不是获利的工具,小作者会一直写下去,希望大家能多多监督我
  • 文章会每攒够两篇进行更新发布(受平台原因,也是希望能让更多的人看见)
  • 感谢各位的阅读希望我的文章会对诸君有所帮助

 

  • 4
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值