C++的内存模型分为四大区
代码区:存放函数体的二进制代码,由操作系统进行管理的
全局区:存放全局变量和静态变量以及常量
栈区:由编译器自动分配释放,存放函数的参数值,局部变量等
堆区:由程序员分配和释放,若程序员不释放,程序结束时由操作系统回收
一. 代码区
存放CPU执行的机器指令
代码区是共享的,共享的目的是对于频繁被执行的程序,只需要在内存中有一份代码即可
代码区是只读的,使其只读的原因是防止程序意外修改了它的指令
二. 全局区
全局变量和静态变量存放在此
全局变量还包含了常量区,字符串常量和其他常量也存放在此
该区域的数据在程序结束后由操作系统释放
int g_a = 10; // 放在全局区
const int c_g_a = 10; // 放在全局区
int main()
{
int a = 10; // 不存放在全局区
static int s_a = 10; // 放在全局区
const int c_a = 10; // 不放在全局区
string str = "haiyang";
cout << (int)&g_a << endl; // 全局区
cout << (int)&a << endl; // 不是全局区
cout << (int)&s_a << endl; // 全局区
cout << (int)&c_a << endl; // 不是全局区
cout << (int)&str << endl; // 不是全局区
cout << (int)&"haiyang" << endl; // 全局区
cout << (int)&c_g_a << endl; // 全局区
return 0;
}
三. 栈
由编译器自动分配释放,存放函数的参数值,局部变量等
注意事项:不要返回局部变量的地址,栈区开辟的数据由编译器自动释放
int* func() {
//形参数据也会放在栈区
int a = 10;//局部变量,存放在栈区,栈区的数据在函数执行后自动释放
return &a;//返回局部变量的地址
}
int main() {
//接受func函数的返回值
int* p = func();
cout << *p << endl;//第一次可以打印正确的数字是因为编译器做了保留
cout << *p << endl;//第二次这个数据就不会在保留了
system("pause");
return 0;
}
四. 堆
由程序员分配释放,若程序员不释放,程序结束时由操作系统回收
在c++中主要利用new在堆区开辟内存
int* func() {
//利用new关键字,可以将数据开辟到堆区
//指针 本质也是局部变量,放在栈上,指针保存的数据是放在堆区
int* p = new int(10);
return p;
}
int main() {
//在堆区开辟数据
int* p = func();
cout << *p << endl;
cout << *p << endl;
cout << *p << endl;
system("pause");
return 0;
}
五. new关键词
int* func1() {
// 在堆区创建整型数据,返回的是该数据的地址,可以用指针接收
int* p = new int(10);
return p;
}
void test01() {
int* p = func1();
// 因为是在堆里的数据,所以不会因为函数执行完毕被释放
cout << *p << endl;
//堆区的数据由程序员开辟和释放,如果想释放可以利用关键字delete关键字
delete p;
cout << *p <<endl; // 内存已经被释放,再次访问就属于非法操作
}
void test02() {
//创建10个整型数据的数组,在堆区
int* arr = new int[10];//10代表有10个元素
for (int i = 0; i < 10; i++) {
arr[i] = i + 100;//给十个元素赋值100到109
}
for (int i = 0; i < 10; i++) {
cout << arr[i] << endl;
}
//释放数组的时候,要加[]才可以
delete[] arr;
}
int main() {
test01();
test02();
system("pause");
return 0;
}