【C语言复习(二十七)】野指针和内存操作分析

1、初识野指针

        野指针和内存操作分析野指针通常是因为指针变量中保存的值不是一个合法的内存地址而造成的;
        野指针不是NULL指针,是指向不可用内存的指针;

        NULL指针不容易用错,因为if语句很好判断一个指针是不是NULL;

2、野指针的由来

        局部指针变量没有被初始化;

#include <stdio.h>
#include <string.h>

struct Student
{
    char* name;
    int number;
};

int main()
{
    struct Student s;

    strcpy(s.name, "Hello Body"); // OOPS!

    s.number = 99;

    printf("%s\n",s.name);
    return 0;
}

        使用已经释放过后的指针;

#include <stdio.h>
#include <malloc.h>
#include <string.h>

void func(char* p)
{
    printf("%s\n", p);
    free(p);
}

int main()
{
    char* s = (char*)malloc(5);
    
    strcpy(s, <span style="font-family: Arial, Helvetica, sans-serif;">"Hello Body"</span>);
    
    func(s);
    
    printf("%s\n", s); // OOPS!
    
    return 0;
}

        指针指向的变量在指针之前被销毁;

#include <stdio.h>

char* func()
{
    char p[] = <span style="font-family: Arial, Helvetica, sans-serif;">"Hello Body"</span><span style="font-family: Arial, Helvetica, sans-serif;">;</span>
    
    return p;
}

int main()
{
    char* s = func();
    
    printf("%s\n", s); // OOPS!
    
    return 0;
}

3、常见错误分析

3.1 非法内存操作

结构体成员指针未初始化
没有为结构体指针分配足够的内存空间

#include <stdio.h>
#include <malloc.h>


struct Demo
{
    int* p;
};


int main()
{
    struct Demo d1;
    struct Demo d2;
    
    int i = 0;
    
    for(i=0; i<10; i++)
    {
        d1.p[i] = 0; // OOPS!
    }
    
    d2.p = (int*)calloc(5, sizeof(int));
    
    for(i=0; i<10; i++)
    {
        d2.p[i] = i; // OOPS!
    }
    
    free(d2.p);
    
    return 0;
}

3.2 内存初始化分析

内存分配成功,但并未初始化

#include <stdio.h>
#include <malloc.h>

int main()
{
    char* s = (char*)malloc(10);
    
    printf(s); // OOPS!
    
    free(s);
       
    return 0;
}

3.3 内存越界

数组越界

#include <stdio.h>

void f(int a[10])
{
    int i = 0;
    
    for(i=0; i<10; i++)
    {
        a[i] = i; // OOPS!
        printf("%d\n", a[i]);
    }
}

int main()
{
    int a[5];
    
    f(a);
       
    return 0;
}

3.4 内存泄露

#include <stdio.h>
#include <malloc.h>

void f(unsigned int size)
{
    int* p = (int*)malloc(size*sizeof(int));
    int i = 0;
    
    if( size % 2 != 0 )
    {
        return; // OOPS!
    }
    
    for(i=0; i<size; i++)
    {
        p[i] = i;
        printf("%d\n", p[i]);
    }
    
    free(p);
}

int main()
{
    f(9);
    f(10);
       
    return 0;
}

3.5 多次指针释放

#include <stdio.h>
#include <malloc.h>

void f(int* p, int size)
{
    int i = 0;
    
    for(i=0; i<size; i++)
    {
        p[i] = i;
        printf("%d\n", p[i]);
    }
    
    free(p);
}

int main()
{
    int* p = (int*)malloc(5 * sizeof(int));
    
    f(p, 5);
    
    free(p); // OOPS!
       
    return 0;
}

3.6使用已经释放的指针

#include <stdio.h>
#include <malloc.h>

void f(int* p, int size)
{
    int i = 0;
    
    for(i=0; i<size; i++)
    {
        printf("%d\n", p[i]);
    }
    
    free(p);
}

int main()
{
    int* p = (int*)malloc(5 * sizeof(int));
    int i = 0;
    
    f(p, 5);
    
    for(i=0; i<5; i++)
    {
        p[i] = i; // OOPS!
    }
       
    return 0;
}

4、错误防范准则

        使用malloc申请了内存后,应该立即检查指针值是否为NULL,防止使用值为NULL的指针;
        牢记数组长度,防止数组越界操作,考虑使用柔性数组;

        动态申请操作必须和释放操作匹配,防止内存泄露和多次释放;

        free指针之后必须立即赋值为NULL,防止误操作;

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值