当前位置: 首页 > news >正文

对于循环的一次探索

写算法题的时候突然想到,为什么循环内定义变量不会算作是重复定义,以及变量作用域问题,为此,进行了一次探索


c:

代码:

#include <stdio.h>
int main()
{int a = 0;int* b = &a;for (int i = 0; i < 3; i++) {printf("%d\n", a);printf("%p\n",&a);int a = i;printf("%d\n", a);printf("%p\n", &a);printf("------------------\n");}return 0;
}

运行结果:

0
010FF91C
0
010FF8F8
------------------
0
010FF91C
1
010FF8F8
------------------
0
010FF91C
2
010FF8F8
------------------

java:

代码:

public class Main{public static void main(String[]args) {int a = 1;for(int i = 0 ;i<2;i++){int a = i;System.out.println(a);System.out.println("---------------------");}}
}

java: 已在方法 main(java.lang.String[])中定义了变量 a


go:

代码:

import "fmt"func main() {a := 0for i := 0; i < 3; i++ {fmt.Println(a)fmt.Println(&a)var a = ifmt.Println(a)fmt.Println(&a)fmt.Println("________________________")}
}

结果:

0
0xc00000a0b8            
0                       
0xc00000a100            
________________________
0                       
0xc00000a0b8            
1                       
0xc00000a110            
________________________
0                       
0xc00000a0b8            
2                       
0xc00000a120            
________________________

查阅资料得知,这个属于是变量作用域的问题,局部变量如果超出作用域就会销毁,正如C语言代码运行结果那样,循环内定义同一变量的地址值是不会改变的。

而且代码在编译时,会将重复定义的变量”提出“到循环外定义,也就是说,在循环内定义和循环外定义在占用内存上几乎一样

至于同名变量可以既在循环内定义,又在循环外定义,这点还是跟作用域有关,但为什么java不能这样定义,emmm,没有查到相关资料,知道的可以补充一下


还有个问题,那为什么go定义出来的变量地址不一样呢?是没有销毁原来的局部变量吗?这就要说到Go 有 escape analysis, 在编译期,它会分析你的变量是否在函数执行完毕那一刻,程序有没有可能有别的对象引用到它(所谓逃逸),如果没有,那这个变量就可以在栈上分配,完全不经过 gc,如果它已经逃逸了,那就不知道什么时候才能gc了~

这个代码中的逃逸是由于fmt导致的,我们不妨不恰当的修改一下代码

import "fmt"func main() {a := 0for i := 0; i < 3; i++ {println(a)println(&a)var a = iprintln(a)println(&a)fmt.Println("________________________")}
}

结果:

0
0xc00006ff18            
0                       
0xc00006ff10            
________________________
0                       
0xc00006ff18            
1                       
0xc00006ff10            
________________________
0                       
0xc00006ff18            
2                       
0xc00006ff10            
________________________

可以看到,当我们不使用标准输出流,而是使用标准错误流打印的时候,可能就不会触发逃逸,看到此时地址是一样的,也就是印证了我们上面的结论。

http://www.lryc.cn/news/289510.html

相关文章:

  • 设计模式:简介及基本原则
  • 营销领域有哪些著名的模型?如销售漏斗等
  • JavaScript学习-let、var、const的使用
  • 【Java】SpringMVC参数接收(一)
  • File类知识点回顾
  • 2024新版68套Axure RP大数据可视化大屏模板及通用组件+PSD源文件
  • Optional lab: Linear Regression using Scikit-LearnⅠ
  • CentOS使用
  • [SWPUCTF 2018]SimplePHP1
  • api管理工具的新发现
  • 2024 年 eBPF 和网络趋势预测
  • 2024.1.27 GNSS 学习笔记
  • Unity - 将项目转为HDRP
  • ETCD高可用架构涉及常用功能整理
  • 深度学习中RGB影像图的直方图均衡化python代码and对图片中指定部分做基于掩模的特定区域直方图均衡化
  • PyTorch深度学习实战(33)——条件生成对抗网络(Conditional Generative Adversarial Network, CGAN)
  • 编写Bash脚本程序从记录文件中提取history命令的优化,再介绍linux bash语法和结构
  • Python中Numba库装饰器
  • Spring Boot Aop 执行顺序
  • 100天精通鸿蒙从入门到跳槽——第16天:ArkTS条件渲染使用教程
  • 【Linux C | 进程】Linux 进程间通信的10种方式(1)
  • 橘子学Mybatis08之Mybatis关于一级缓存的使用和适配器设计模式
  • 看图说话:Git图谱解读
  • linux新增用户,指定home目录和bash脚本且加入到sudoer列表
  • 经典目标检测YOLO系列(三)YOLOV3的复现(1)总体网络架构及前向处理过程
  • OpenGL/C++_学习笔记(四)空间概念与摄像头
  • C语言2024-1-27练习记录
  • 深入解析HTTPS:安全机制全方位剖析
  • 【197】JAVA8调用阿里云对象存储API,保存图片并获取图片URL地址。
  • 2024.1.24 C++QT 作业