传递动态内存

一、内存分配分类

1.从静态存储区域分配。内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在。例如全局变量,static 变量。

2.在栈上创建。在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。

3.从堆上分配,亦称动态内存分配。程序在运行的时候用malloc new 申请任意多少的内存,程序员自己负责在何时用free delete 释放内存。动态内存的生存期由程序员决定,使用非常灵活,但问题也最多。

二、传递动态内存

1.动态指针传递失败

 1 #include<iostream>
 2 using namespace std;
 3 void GetMe(char *p,int num)
 4 {
 5     p=(char*)malloc(sizeof(char)*num);    
 6 }
 7 int main()
 8 {
 9     char *str=NULL;
10     GetMe(str,100);
11     strcpy(str,"Hello!");
12     cout<<str<<endl;
13     return 0;
14 }

 

结果:程序运行奔溃,因为str还是为NULL,往空的地址强行赋值时,内存出错

原因:str并没有获取指针p开辟的空间。本质为调用函数 GetMe 时,函数会初始化函数内的局部变量,同时为传进来的实参str(指针和值都创建,引用除外)创建一个副本 _p,  _num,所以 p申请了内存,只是把p指向的内存地址改变,而str并没有改变,所以str依然没有获得内存。同时每次p申请的内存都不会得到释放,最终会造成内存泄露。

2.正确的传递动态内存

2.1  返回指针

 1 #include<iostream>
 2 using namespace std;
 3 char* GetMe(char *p,int num)
 4 {
 5     p=(char*)malloc(sizeof(char)*num);
 6     return p;
 7 }
 8 int main()
 9 {
10     char *str=NULL;
11     str=GetMe(str,100);
12     strcpy(str,"Hello!");
13     cout<<str<<endl;
14     delete str;
15     return 0;
16 }

结果:正常运行

原因:p申请了空间,并将该空间的地址作为返回值传给str,这样str就指向了p申请的内存空间

2.2 传递指针

 1 #include<iostream>
 2 using namespace std;
 3 void GetMe(char **p,int num)
 4 {
 5     *p=(char*)malloc(sizeof(char)*num);    
 6 }
 7 int main()
 8 {
 9     char *str=NULL;
10     GetMe(&str,100);
11     strcpy(str,"Hello!");
12     cout<<str<<endl;
13     delete str;
14     return 0;
15 }

结果:正常运行

原因:传递了str的指针给函数GetMe(),那么p就是str的地址的副本,地址的副本指向的内存是固定,所以该函数是为str地址指向的str开辟空间

 3.引用传递

 1 #include<iostream>
 2 using namespace std;
 3 void GetMe(char* &p,int num)
 4 {
 5     p=(char*)malloc(sizeof(char)*num);    
 6 }
 7 int main()
 8 {
 9     char *str=NULL;
10     GetMe(str,100);
11     strcpy(str,"Hello!");
12     cout<<str<<endl;
13     return 0;
14 }

结果:正常运行

原因:引用就是原传递实参的别名,地址相同,指向相同的地址空间

 

总结:实际上指针传递仍然是一种值传递,只不过在参数是指针的时候,传递的是指针的副本,这样在地址上的操作实际就反映到了内存中

 

 

 

转载于:https://www.cnblogs.com/Dana-gx/p/9741225.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/250561.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

linux --- 基础指令

基础命令 1、ls(list) 用法1: # ls 含义: 列出当前工作目录下所有的 文件/文件夹 的名称 用法2: # ls 路径 含义: 列出指定路径目录下所有的 文件/文件夹 的名称 用法3: # ls 选项 路径 含义: 以指定的格式来显示指定目录下文件夹的名称 栗子: # ls -l 路径 -->> 表…

验证码功能

验证码功能 1.安装captcha插件 (dj_login) D:\dj\dj_login>pip install django-simple-captcha Collecting django-simple-captchaUsing cached https://files.pythonhosted.org/packages/d7/f4/ea95b04ed3abc7bf225716f17e35c5a185f6100db4d7541a 46696ce40351/django-simp…

Java 类的成员

Java 类的成员 初始化块 1、一个类中初始化块若有修饰符&#xff0c;则只能被static修饰&#xff0c;称为静态代码块(staticblock )&#xff0c;当类被载入时&#xff0c;类属性的声明和静态代码块先后顺序被执行&#xff0c;且只被执行一次。 2、static块通常用于初始化sta…

linux --- 进阶指令

进阶指令(重点) 1、df 指令 作用: 查看磁盘空间语法: # df -h 注: -h:以较高可读性的方式展示出来 2、free 指令 作用: 查看内存使用情况语法: # free -m 注: -m:以M的单位显示内存情况 -/ buffers/cache: free 代表真实可用的内存为 486 Mb Swap: 表示,临时将硬盘当作内存…

MFC对话框播放8位512*512的像素数据

关键代码&#xff1a; UINT playAllFrame(LPVOID lpParameter){//showOneFrame(0,TRUE);CMFCDialogDlg *mydlg (CMFCDialogDlg *) lpParameter;//获取原始数据文件CString selectPath;mydlg->GetDlgItemTextW(IDC_MFCEDITBROWSE,selectPath);string StrSelectPath(CW2A(sel…

java 集合 CopyOnWriteArrayList

CopyOnWriteArrayList 也是实现List接口他是在concurrent 包里面&#xff0c;所以他是线程安全的&#xff0c;其他的基本和ArrayList很想。他线程安全是用ReentrantLock 实现的&#xff0c;他内部有一个ReentrantLock对象&#xff0c;然后在增删改的时候都操作这个锁对象&#…

Java 类的特性1

Java 类的特性1 继承 1.为什么要有继承&#xff1f; 多个类中存在相同属性和行为时&#xff0c;将这些内容抽取到单独一个类中&#xff0c;那么多个类无需再定义这些属性和行为&#xff0c;只要继承那个类即可。 2.此处的多个类称为子类&#xff0c;单独的这个类称为父类&a…

linux --- 高级指令

高级指令 1、hostname 指令 作用: 操作(读取|操作)服务器的主机名语法1: # hostname (输出完整的主机名) 语法2: # hostname -f (输出当前主机中的FQDN) FQDN&#xff1a;(Fully Qualified Domain Name)全限定域名&#xff1a;同时带有主机名和域名的名称。 2、id 指令 作…

Linux修改密码后不能SSH远程登录了

1、把以下文件的属性改成755&#xff0c;然后再修改密码&#xff1a;/etc/passwd ,/etc/group , /etc/shadow , /etc/gshadow2、如果文件的属性无法更改&#xff0c;请用lsattr 查看文件是否有 i 属性&#xff0c;如有&#xff0c;则用chattr取消之&#xff0c;如&#xff1a;l…

Java 类的特性2

Java 类的特性2 类属性、类方法的设计思想 类属性作为该类各个对象之间共享的变量。在设计类时,分析哪些类属性不因对象的不同而改变&#xff0c;将这些属性设置为类属性。相应的方法设置为类方法。如果方法与调用者无关&#xff0c;则这样的方法通常被声明为类方法&#xff…

docker --- 镜像、容器

Docker使用国内的源 windows下使用 "everything"软件 查找 daemon.json修改为如下: {"registry_mirrors": ["https://docker.mirrors.ustc.edu.cn"] }查看images(镜像) docker images注: 看见的镜像是已经下载好的,因此在没有网络的情况下也可…

Java 面向对象

Java 面向对象 面向对象的三大特征 封装 (Encapsulation)继承 (Inheritance)多态 (Polymorphism) 类的访问机制&#xff1a; 在一个类中的访问机制&#xff1a;类中的方法可以直接访问类中的成员变量。&#xff08;例外&#xff1a;static方法访问非static&#xff0c;编译…

交换机老化测试和性能测试方法收集

说明&#xff1a;这是一个做交换机朋友给的一个方法&#xff0c;只做老化测试&#xff0c;不做压力满载测试。 我所理解的&#xff1a;老化测试是指在一定的时间内工作负荷之后&#xff0c;看有没有故障和不稳定的现象出现。 而对于压力测试&#xff0c;需要通过专业的测试设备…

【webGL入门2】点线面的绘制

用js绘制webGL的点&#xff1a; THREE.Vector3 function ( x, y, z ) {    //用THREE声明的变量都是全局变量。this.x x || 0;this.y y || 0;this.z z || 0;}; 注意&#xff1a;“||”&#xff08;或&#xff09;运算符&#xff0c;就是当xnull或者undefine时&#xff…

docker --- mysql的部署

MySQL部署 [1]查询本地镜像中是否含有 centos/mysql-57-centos7(我们用到的镜像) docker images[2] 拉取镜像 docker pull centos/mysql-57-centos7注: centos/mysql-57-centos7 是我们用到的镜像 [3] 创建容器 docker run -id --nametensquare_mysql -p 33306:3306 -e M…

Java中的排序

Java中的排序 排序方法的选择 1.若n较小(如n≤50)&#xff0c;可采用直接插入或直接选择排序。当记录规模较小时&#xff0c;直接插入排序较好&#xff1b;否则因为直接选择移动的记录数少于直接插入&#xff0c;应选直接选择排序为宜。 2.若文件初始状态基本有序(指正序)&a…

Codeforces Round #493 (Div. 2) C. Convert to Ones 乱搞_构造_好题

题意&#xff1a; 给你一个长度为 nnn 的 010101串 &#xff0c;你有两种操作&#xff1a; 1.将一个子串翻转&#xff0c;花费 XXX 2.将一个子串中的0变成1&#xff0c;1变成0&#xff0c;花费 YYY 求你将这个01串变成全是1的串的最少花费。 首先&#xff0c;我们可以将串按照0…

[T-ARA][그녀를 보면][看着那个女人的话]

歌词来源&#xff1a;http://music.163.com/#/song?id29343995 作曲 : 코난 [作曲 : Ko-nan] 作词 : 코난/로코 [作词 : Ko-nan-/lo-Ko] baby i hate you [baby i hate you] but i love you [but i love you] cant live without you [cant live without you] baby i hate you …

node --- 连接mysql(docker环境) Sequelize库

mysql 数据库 [1] 首先配置 docker 环境 采用 docker-compose 方法 源码: /test-mysql/docker-compose.yml version: 3.1 services:mysql:image: mysqlcommand: --default-authentication-pluginmysql_native_passwordrestart: alwaysenvironment:MYSQL_ROOT_PASSWORD: examp…

Java-接口练习

Java-接口练习 编写2个接口&#xff1a;InterfaceA和InterfaceB&#xff1b;在接口InterfaceA中有个方法voidprintCapitalLetter()&#xff1b;在接口InterfaceB中有个方法void printLowercaseLetter()&#xff1b;然 后写一个类Print实现接口InterfaceA和InterfaceB&#xff0…