dreamwearver可以做网站吗/百度推广管理

dreamwearver可以做网站吗,百度推广管理,中国十大网站建设企业,郑州网约车资格证网上报名在上一章中 C 语法之 指针的一些应用说明-CSDN博客 我们了解了指针变量&#xff0c;int *p;取变量a的地址这些。 那么函数同样也有个地址&#xff0c;直接输出函数名就可以得到地址&#xff0c;如下&#xff1a; #include<iostream> using namespace std; void fun() …

在上一章中 C++ 语法之 指针的一些应用说明-CSDN博客

我们了解了指针变量,int *p;取变量a的地址这些。

那么函数同样也有个地址,直接输出函数名就可以得到地址,如下:

#include<iostream>
using namespace std;
void fun()
{int a = 5;}
int main()
{cout << "fun函数地址:" << fun;
}

返回函数地址:

这个就是系统为这个函数代码分配的内存空间的首地址。

既然有这个内存地址,那相对的,跟变量指针一样,也有函数指针,我们要怎么定义函数指针变量呢?

像int *p; char *cptr; 可以看到,定义指针变量需要提前进行类型区分。

所以函数指针,也是如此,必须说明这个函数的返回值,以及参数类型,几个参数。

如下定义:

void (*pFun)();

如果fun函数是这样

void fun(int a);

则对应的函数指针:

void (*pFun)(int);

你们可以在编译器尝试一下,是必须对应一致的,函数指针不能指向定义不一样的函数。

就像int *p指针不能指向char变量地址一样,只能是int类型的。

现在我们来一个例子学习一下:
 

#include<iostream>
using namespace std;
void (*pFun)();
void fun1()
{cout << "我是fun2";
}void fun()
{cout << "我是fun\n";pFun = fun1; //在这里 函数指针 指向另一个函数 pFun;
}int main()
{pFun = &fun; //为了便于阅读用&fun, 事实上直接pFun=fun也可以,下面加*号同理  可省略(*pFun)();(*pFun)();
}

结果: 

上面的例子说明了,pFun指针指向fun,就是fun函数,指向fun1就是fun1函数。 

其实跟变量指针一个道理。

函数的生命周期 

我们知道在函数内正常定义的变量叫局部变量,函数执行完了,这些变量的内存空间就被释放了。

也就是函数每执行一次,比如有个int a=5;就会给a分配内存空间,执行完了,就会释放。

这种由系统分配,系统释放的内存空间,就是内存中的栈空间。

也就是说,这种变量在栈中申请空间。由系统管理释放。

而堆中,是由程序员自己申请的内存空间:

比如C的malloc函数(需free释放)

C++的new 

例: int* ptr = new int; (需delete释放)

像这些,申请的空间,系统不会帮你自动释放,所以就需要你自己手动释放,否则,这块空间即使不使用了,也会被程序一直占用。它并没有栈中内存空间的功能,自动释放。

这个内存堆栈空间是系统定义的,而物理上内存并没有此种划分,需要明白。

下面我们用一些例子来证明:

#include <iostream>using namespace std;
int i = 0;  
void fun() {int a = 5;cout << "第" << i << "次地址:" << &a << endl;
}int main() {for (; i < 4; i++) {fun();}return 0;
}

每次调用fun函数,输出局部变量a的地址,理论上应该是不同的地址,因为每次函数调用完之后a变量在栈中的空间会被释放。

但是结果:

地址是一样的,这是系统优化分配的原因,因为这个地址被释放了,下次分配还可以找同样的地址。这是合法的,就像在磁盘删掉一个文件,然后再存储,还是原来的位置。

所以由于这种现象,上面这个代码并不能证明变量被重新分配内存空间。

我们要怎么做,在函数执行期间,调用其它函数干扰栈空间分配,就像磁盘删除文件,然后复制大量其它文件,这样再粘贴的文件位置就会不一样?

比如下面这个:

#include <iostream>using namespace std;
int i = 0;
void fun() {int a = 5;cout << "第" << i << "次地址:" << &a << endl;
}
void other()
{int b = 5;int c = 6;int bc[56] = { 0 };
}
int main() {for (; i < 4; i++) {fun(); other();//调用一下其它函数,里面申请栈空间,打乱分配。}return 0;
}

并没有用,原因是other里的也是局部变量,执行完后,同样变量占用的栈空间也被释放了。所以跟原来的还是一样。 

那我们想个方法,不被释放,看下面的代码:

#include <iostream>using namespace std;
int i = 0;
void fun(int sum) {int a = 5;cout << "第" << sum << "次地址:" << &a << endl;
}int main() {for (; i < 4; i++) {fun(i);int b = 5; //定义 b变量,然后再调用fun(11) 此时b变量和fun(11)同时在for的作用域中fun(11);int c = 6;fun(22);}return 0;
}

很遗憾,还是没有效果,所有的a变量地址都是一样的。

我分析可能是进入for作用域一次性的分配好了(有待验证)。

从这个现象可以看出,虽然每次执行函数时其中的局部变量,都是重新分配,但系统遵循着某一种优化规则,使得每次分配的地址尽可能一样。

由于方向上的问题,这个规则就不深入研究了。

好了,通过直观的查看地址方法已失败,实验起来比较困难。

我们可以从侧面来验证,有两个方法,第一个通过值的变化来验证,如果内存空间被释放了,那么的它的值如果没有保留,那可以证明函数执行完,局部变量已经被释放。

取值验证:

#include <iostream>using namespace std;
int i = 0;
int* ptr;
void fun(int sum) {int a = 5;ptr = &a; //将a变量的地址存到全局指针变量ptr中 以便在函数外访问cout << "\n fun函数内a值:" << *ptr;
}
int main() {for (; i < 2; i++) {fun(i); cout << "\n函数外的a值:" << *ptr;}return 0;
}

结果:

可以看到,同样是取*ptr的值,函数外已经变了,说明系统没有为变量a留有内存空间来保存值了,函数执行完就被释放了。 

第二个通过递归调用函数的方法强制验证,这样的地址绝对不能相同,比如说递归调用4次函数。

局部变量a肯定是不同的地址,如果每一次都重新分配空间的话。

为什么,因为在递归未完成时,所有的局部变量都不会被释放。因为所有的函数都没执行完。

它想复用上一次变量A的地址是不可能的。

这个方法是反向证明,证明每次是重新分配空间的。然后就可以佐证,即然每次执行重新分配空间,那么执行完了也应当是释放空间的。

递归调用验证:

#include <iostream>using namespace std;void fun(int sum) {int a = 5;if (sum == 0) return; //用sum来控制 递归调用fun函数,防止无限循环调用else{cout << "\nsum=" << sum << "时,a的地址:" << &a;sum = sum - 1;fun(sum);//递归调用fun}}
int main() {fun(4);return 0;
}

结果:

可以看到,每次a变量地址是不同的,四个不同的地址。 

说了局部变量,这里有个有趣的点,有没有一种变量,我不想每次函数执行,重新分配和释放,是一直存在的,有,就是在函数内被static修饰的变量,这种变量跟全局变量一样,它的空间不是在堆栈中,而是静态内存空间中,从整个程序开始分配,运行期间一直存在,到程序结束才释放。

 代码示例:

using namespace std;
int g = 5;
void fun() {static int s = 5; //静态变量int b = 10;int c = 10;s++;cout << "\n静态s变量值:" << s << "------地址:" << &s << "-----全局变量g地址:" << &g;cout << "\n局部变量b地址:" << &b << "---局部变量c地址:" << &c;}
int main() {for(int i=0;i<3;i++)fun();return 0;
}

运行结果:

 1.从地址分配来看,可以证明,全局变量g和静态变量s 的地址相近,说明它们在同一块内存区域(静态存储区)。有着相同的特性。

2.而局部变量b,c又是另一块内存区域(动态存储区),即栈中。所以它们的地址很接近,只是后几位不同。

3.可以看到静态变量s的值在增长,说明并没有被释放,而开头一句static int s=5;静态变量在定义时赋值只会初始化一次。

 4.另:还记得我们在文章开头取了一个函数地址吗,那么这个属于什么区域呢?这个是代码区,因为函数的执行代码是存储在程序代码区。

这就是一个函数的内存分布区域,不是所有的内容都是一起的。

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

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

相关文章

关于“碰一碰发视频”系统的技术开发文档框架

以下是关于“碰一碰发视频”系统的技术开发文档框架&#xff0c;涵盖核心功能、技术选型、开发流程和关键模块设计&#xff0c;帮助您快速搭建一站式解决方案 --- 随着短视频平台的兴起&#xff0c;用户的创作与分享需求日益增长。而如何让视频分享更加便捷、有趣&#xff0c…

基于django+vue的购物商城系统

开发语言&#xff1a;Python框架&#xff1a;djangoPython版本&#xff1a;python3.8数据库&#xff1a;mysql 5.7数据库工具&#xff1a;Navicat11开发软件&#xff1a;PyCharm 系统展示 系统首页 热卖商品 优惠资讯 个人中心 后台登录 管理员功能界面 用户管理 商品分类管理…

Ardunio 连接OLED触摸屏(SSD1106驱动 4针 IIC通信)

一、准备工作 1、硬件 UNO R3 &#xff1a;1套 OLED触摸屏&#xff1a;1套 导线诺干 2、软件 arduino 二、接线 UNO R3OLED5VVCCGNDGNDA5SCLA4SDA 脚位如下图所示&#xff1a; Uno R3脚位图 触摸屏脚位图 查阅显示屏的驱动规格&#xff1a;通常显示屏驱动芯片有SSD1306,SH110…

07. 面向对象高级(2)_设计模式

什么是设计模式 一个问题通常有种解法&#xff0c;其中肯定有一种解法是最优的&#xff0c;这个最优的解法被人总结出来了&#xff0c;称之为设计模式。 设计模式有20多种&#xff0c;对应20多种软件开发中会遇到的问题。 关于设计模式&#xff0c;主要学什么&#xff1f; 解…

计算机操作系统和进程

目录 一. 操作系统 1. 操作系统的概念 2. 操作系统的功能 二. 进程 1. 进程的概念 2. 进程在系统中的管理 3. PBC &#xff08;1&#xff09;pid &#xff08;2&#xff09;内存指针 &#xff08;3&#xff09;文件标识符 4. 资源分配 5. 进程的调度 &#xff08;…

【Matlab GUI】封装matlab GUI为exe文件

注&#xff1a;封装后的exe还是需要有matlab环境才能运行 &#xff08;1&#xff09;安装MCRinstaller.exe文件&#xff0c;在matlab安装目录下的toolbox/compiler/deploy/win64文件夹里 &#xff08;2&#xff09;安装完MCRinstaller.exe&#xff0c;字命令窗口输入&#x…

登山第二十梯:无人机实时自主探索——我是一只小小小鸟

文章目录 一 摘要 二 资源 三 内容 一 摘要 自主探索是无人机 &#xff08;UAV&#xff09; 各种应用的基本问题。最近&#xff0c;基于 LiDAR 的探索因其能够生成大规模环境的高精度点云地图而受到广泛关注。虽然点云本身就为导航提供了信息&#xff0c;但许多现有的勘探方…

JAVA序列化与反序列化URLDNS链CC1链

1、序列化的实现 java序列化的是对象属性的&#xff0c;只有实现了Serializable或者Externalizable接口的类的对象才能被序列化为字节序列。&#xff08;不是则会抛出异常&#xff09;&#xff0c;静态成员变量是属于类的&#xff0c;所以静态成员变量是不能被序列化的&#x…

Android7 Input(一)Android Input服务初始化

本系列博客主要描述Android 7.1系统中输入管理服务InputManagerService的源码分析。 概述 本文主要描述了InputManagerService服务的初始化和启动&#xff0c;在Android7系统上InputManagerService服务的框架如下所示: 注&#xff1a;箭头的方向&#xff0c;并不能真实代表数…

用于 RGB-D 显著目标检测的点感知交互和 CNN 诱导的细化网络(问题)

摘要 问题一&#xff1a;但在对自模态和跨模态的全局长距离依赖关系进行建模方面仍显不足。什么意思&#xff1f; 自模态&#xff08;Intra-modal&#xff09;全局依赖&#xff1a;在同一模态内&#xff0c;长距离像素之间的信息交互对于理解全局背景很重要&#xff0c;但 CN…

html5基于Canvas的经典打砖块游戏开发实践

基于Canvas的经典打砖块游戏开发实践 这里写目录标题 基于Canvas的经典打砖块游戏开发实践项目介绍技术栈核心功能实现1. 游戏初始化2. 游戏对象设计3. 碰撞检测系统4. 动画系统5. 用户界面设计 性能优化1. 渲染优化2. 内存管理 项目亮点技术难点突破项目总结 项目介绍 在这个…

MySQL外键约束下的索引删除难题:从报错到完美解决的实战指南

&#x1f6a8; MySQL外键约束下的索引删除难题&#xff1a;从报错到完美解决的实战指南 &#x1f525; 问题背景&#xff1a;一个看似简单的删除操作引发的连环坑 场景复现&#xff1a;某日接到需求&#xff0c;需删除 invite_codes 表中的冗余索引 FKnqn27fyjlgio5y60eieohi0…

Rust嵌入式开发环境搭建指南(基于Stm32+Vscode)

Rust嵌入式开发环境搭建指南(基于Stm32+Vscode) 部分目录如下所示: 目录 简介Rust开发环境安装STM32开发工具链安装VSCode环境配置VSCode插件安装调试器配置项目创建与配置常见问题与解决方案简介 本文档旨在指导开发者如何搭建基于Rust语言的STM32嵌入式开发环境。相比传…

动态规划合集——动态规划基本原理

动态规划合集——动态规划基本原理 动态规划原理1258&#xff1a;【例9.2】数字金字塔 动态规划原理深度优先搜索记忆化搜索动态规划&#xff08;顺推&#xff09;动态规划原理题解分析 滚动数组优化动态规划&#xff08;逆推&#xff09; 动态规划原理 从数塔问题出发理解动态…

如何让节卡机器人精准对点?

如何让节卡机器人精准对点&#xff1f; JAKA Zu 软件主界面主要由功能栏、开关栏、菜单栏构成。 菜单栏&#xff1a;控制柜管理&#xff0c;机器人管理与软件管理组成。主要功能为对控制柜关机、APP 设置、机器人本体设 置、控制柜设置、连接机器人和机器人显示等功能。 开关…

实时视频分析的破局之道:蓝耘 MaaS 如何与海螺 AI 视频实现高效协同

一、蓝耘 MaaS 平台&#xff1a;AI 模型全生命周期管理的智能引擎 蓝耘 MaaS&#xff08;Model-as-a-Service&#xff09;平台是由蓝耘科技推出的 AI 模型全生命周期管理平台&#xff0c;专注于为企业和开发者提供从模型训练、推理到部署的一站式解决方案。依托云原生架构、高…

设计模式(行为型)-策略模式

目录 定义 类图 角色 角色详解 Strategy&#xff08;抽象策略类&#xff09;​ Context&#xff08;环境类 / 上下文类&#xff09;​ ConcreteStrategy&#xff08;具体策略类&#xff09;​ 优缺点 优点​ 缺点​ 使用场景 类行为差异场景​ 动态算法选…

【算法day14】三数之和

三数之和 https://leetcode.cn/problems/3sum/description/ 给你一个整数数组 nums &#xff0c;判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i ! j、i ! k 且 j ! k &#xff0c;同时还满足 nums[i] nums[j] nums[k] 0 。请你返回所有和为 0 且不重复的三元组。…

10、STL中的unordered_map使用方法

一、了解 1、unordered_map(哈希) unordered_map是借用哈希表实现的关联容器。 访问键值对O&#xff08;1&#xff09;&#xff0c;最坏情况O&#xff08;n&#xff09;&#xff0c;例如哈希冲突严重时。【n是一个哈希桶的元素数量】 unordered_map特性 键值对存储&#xff…

C++ 头文件说明

如果一个程序足够大&#xff0c;代码功能很多&#xff0c;可以想象&#xff0c;不可能把代码写在一个cpp文件里。我们需要模块化&#xff0c;这样的好处很多&#xff0c;方便分工合作&#xff0c;可读性提高&#xff0c;调用也方便。 这个要怎么做呢&#xff1f; 很简单直接当…