int val = 20;//在栈空间上开辟四个字节
char arr[10] = {
0};//在栈空间上开辟10个字节的连续空间
但是上述的开辟空间的方式有两个特点:
1. 空间开辟大小是固定的。
2. 数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配。
但是对于空间的需求,不仅仅是上述的情况。有时候我们需要的空间大小在程序运行的时候才能知道,那数组的编译时开辟空间的方式就不能满足了。 这时候就只能试试动态存开辟了。
函数参数:
函数返回值:
函数的使用:
#include
#include
#include
int main()
{
//开辟10个整型空间,并用ptr管理
int* ptr = (int*)malloc(sizeof(int) * 10);
//但也许内存不足开辟失败,需要判断一下是否开辟成功
if (ptr == NULL)
{
//失败,打印错误信息,并返回-1(非正常结束)
printf("malloc fail!\n");
printf("%s\n", strerror(errno));//也可使用系统错误码打印错误信息,但需包含头文件
return -1;
}
return 0;
}
C语言提供了另外一个函数free,专门是用来做动态内存的释放和回收的
free函数用来释放动态开辟的内存。
如上述代码,需要在函数结束前(或者你想要释放的时候)使用free函数
free(ptr);//释放空间
ptr=NULL;//将指针置空,不然为野指针
int main()
{
int* ptr = (int*)calloc(10, sizeof(int));
if (ptr == NULL)
{
//失败
printf("malloc fail!\n");
return -1;
}
for (int i = 0; i < 10; i++)
{
printf("%d ", *(ptr + i));
}
//记得释放
free(ptr);
ptr = NULL;
return 0;
}
还有一个用来动态内存分配的函数
#include
int main()
{
//开辟10个整型大小空间
int* ptr = (int *)malloc(sizeof(int)*10);
if (ptr == NULL)
{
exit(-1);//exit(0)表示正常退出程序,非零表示非正常退出
}
//扩展容量
//代码1
ptr = realloc(ptr, 1000);//这样可以吗?(如果申请失败会如何?)
//如果申请失败,返回NULL,这个ptr就被赋值为NULL
//原来空间就找不到了,造成内存泄露
//代码2
int* p = NULL;
p = realloc(ptr, 1000);
if (p != NULL)
{
//扩容成功,可以将新的空间交给ptr管理,不影响后续使用
ptr = p;
}
else
{
//扩容失败
}
//释放空间
free(ptr);
ptr = NULL;
return 0;
}
void test()
{
int *p = (int *)malloc(INT_MAX/4);
*p = 20;//如果p的值是NULL,就会有问题,需要先判断
free(p);
}
void test()
{
int i = 0;
int *p = (int *)malloc(10*sizeof(int));
if(NULL == p)
{
exit(-1);
}
for(i=0; i<=10; i++)
{
*(p+i) = i;//当i是10的时候越界访问
}
free(p);
}
void test()
{
int a = 10;
int *p = &a;
free(p);//非动态分配空间,不可使用
}
void test()
{
int *p = (int *)malloc(100);
p++;
free(p);//p不再指向动态内存的起始位置,没有完全释放空间
}
void test()
{
int *p = (int *)malloc(100);
free(p);
free(p);//重复释放,发生错误
}
void test()
{
int *p = (int *)malloc(100);
if(NULL != p)
{
*p = 20;
}
//未释放分配空间
}
int main()
{
test();
while(1);
}
忘记释放不再使用的动态开辟的空间会造成内存泄漏。
切记: 动态开辟的空间一定要释放,并且正确释放
野指针:指针指向非本程序所开辟的空间
题目1:
void GetMemory(char *p)
{
p = (char *)malloc(100);
}
void Test(void)
{
char *str = NULL;
GetMemory(str);
strcpy(str, "hello world");
printf(str);
}
p是局部变量,是str的一份拷贝,p被修改并不能改变str。所以str仍然为NULL,strcpy函数报错。
修改:
//传址
//地址的地址为二级指针
void GetMemory(char **p)
{
//解引用找到str,为其开辟空间
*p = (char *)malloc(100);
}
void Test()
{
char *str = NULL;
GetMemory(&str);//传地址
strcpy(str, "hello world");
printf(str);
//释放空间
free(str);
str=NULL;
}
题目2:
char *GetMemory()
{
char p[] = "hello world";
return p;
}
void Test()
{
char *str = NULL;
str = GetMemory();
printf(str);
}
这是属于返回栈空间地址的一类问题
p是局部变量,p在被调用函数结束之后销毁,str接收p的值,但是该地址指向的空间已经不属于程序,此时str属于野指针。最后打印出错。
题目3:
void GetMemory(char **p, int num)
{
*p = (char *)malloc(num);
}
void Test(void)
{
char *str = NULL;
GetMemory(&str, 100);
strcpy(str, "hello");
printf(str);
}
乍一看没问题,其实是最基础,也是问题最大的错误——忘记释放动态开辟内存。
题目4:
void Test(void)
{
char *str = (char *) malloc(100);
strcpy(str, "hello");
free(str);
//此时已经将str指向的空间释放,str为野指针(虽然值没变),它所指向的空间该程序已经不能访问
if(str != NULL)
{
//向不属于本程序的空间强行赋值,发生错误
strcpy(str, "world");
printf(str);
}
}
C/C++程序内存分配的几个区域:
也许你从来没有听说过柔性数组(flexible array)这个概念,但是它确实是存在的。 C99 中,结构中的最后一个元素允许是未知大小的数组,这就叫做『柔性数组』成员。
例如:
struct st_type
{
int i;
int a[0];//柔性数组成员
};
有些编译器会报错无法编译可以改成:
struct st_type
{
int i;
int a[];//柔性数组成员
};
struct st_type
{
int i;//系统默认对齐数-8 int大小-4 对齐数 -4
int a[0];//柔性数组成员,不占结构体大小
};
printf("%d\n", sizeof(type_a));//输出的是4
我们需要为柔性数组动态开辟内存,大小为 结构大小+想要的数组大小
#include
#include
struct st
{
int i;
int arr[];
};
int main()
{
//为柔性数组开辟10个整型大小空间
struct st* ptr = (struct st*)malloc(sizeof(struct st) + 10 * sizeof(int));
if (ptr == NULL)
{
extit(-1);
}
//使用
ptr->i = 10;
for (int i = 0; i < 10; i++)
{
ptr->arr[i] = i;
}
for (int i = 0; i < 10; i++)
{
printf("%d ", ptr->arr[i]);
}
//若需要扩容
//将数组扩容为20个整型大小
struct st* temp = (struct st*)realloc(ptr, sizeof(struct st) + 20 * sizeof(int));
if (ptr == NULL)
{
extit(-1);
}
//使用
//释放
free(ptr);
ptr = NULL;
return 0;
}
当然如果不使用柔性数组我们也能达到类似的效果,但要注意释放动态分配空间的顺序。
#include
#include
struct st
{
int i;
int* arr;//用指针代替
//在默认对齐数8和指针大小为4的情况下,结构体大小为4+4=8
};
int main()
{
//开辟一个结构体
struct st* s1 = (struct st*)malloc(sizeof(struct st));//创建一个结构体实例
//初始化
s1->i = 10;
//开辟10个整型空间,给arr维护
s1->arr = (int*)malioc(sizeof(int) * 10);
if (s1->arr == NULL)
{
//退出
exit(-1);
}
//使用
//扩容只需给s1->arr扩容即可
int* temp = (int*)realloc(s1->arr, sizeof(int) * 20);
if (s1->arr == NULL)
{
//退出
exit(-1);
}
//使用
//释放,要先释放结构体内部指针指向的空间,然后再释放结构体,否则内部指向的空间无法释放
free(s1->arr);
s1->arr = NULL;
free(s1);
s1 = NULL;
return 0;
}
两种方法比较
上述代码1 和 代码2 可以完成同样的功能,但是方法1 的实现有两个好处: