`
什么世道
  • 浏览: 223692 次
  • 性别: Icon_minigender_1
  • 来自: 长沙
社区版块
存档分类
最新评论

小叙Java变量的作用域和生存期问题

阅读更多

    到目前为止,我们使用的所有变量都是在方法main() 的后面被声明。然而,Java 允许变量在任何程序块内被声明。在前面就了解了,程序块被包括在一对大括号中。一个程序块定义了一个作用域(scope )。这样,你每次开始一个新块,你就创建了一个新的作用域。你可能从先前的编程经验知道,一个作用域决定了哪些对象对程序的其他部分是可见的,它也决定了这些对象的生存期。

    大多数其他计算机语言定义了两大类作用域:全局和局部。然而,这些传统型的作用域不适合Java 的严格的面向对象的模型。当然将一个变量定义为全局变量是可行的,但这是例外而不是规则。在Java 中2个主要的作用域是通过类和方法定义的。尽管类的作用域和方法的作用域的区别有点人为划定。因为类的作用域有若干独特的特点和属性,而且这些特点和属性不能应用到方法定义的作用域,这些差别还是很有意义的。

    方法定义的作用域以它的左大括号开始。但是,如果该方法有参数,那么它们也被包括在该方法的作用域中。因此,现在可认为它们与方法中其他变量的作用域一样。

     作为一个通用规则,在一个作用域中定义的变量对于该作用域外的程序是不可见(即访问)的。因此,当你在一个作用域中定义一个变量时,你就将该变量局部化并且保护它不被非授权访问和/或修改。实际上,作用域规则为封装提供了基础。

    作用域可以进行嵌套。例如每次当你创建一个程序块,你就创建了一个新的嵌套的作用域。这样,外面的作用域包含内部的作用域。这意味着外部作用域定义的对象对于内部作用域中的程序是可见的。但是,反过来就是错误的内部作用域定义的对象对于外部是不可见的。

 

// Demonstrate block scope.
class Scope {
public static void main(String args[]) {
int x; // known to all code within main(全局变量)
x = 10;
if(x == 10) { // start new scope
int y = 20; // known only to this block(局部变量)
// x and y both known here.
System.out.println("x and y: " + x + " " + y);
x = y * 2;
}
// y = 100; // Error! y not known here
// x is still known here.
System.out.println("x is " + x);
}
}

 

 

 

     正如注释中说明的那样,在方法main() 的开始定义了变量x,因此它对于main() 中的所有的随后的代码都是可见的。在if程序块中定义了变量y。因为一个块定义一个作用域,y 仅仅对在它的块以内的其他代码可见。这就是在它的块之外的程序行y=100; 被注释掉的原因。如果你将该行前面的注释符号去掉,编译程序时就会出现错误,因为变量y在它的程序块之外是不可见的。在if程序块中可以使用变量x,因为块(即一个嵌套作用域)中的程序可以访问被其包围作用域中定义的变量。

     变量可以在程序块内的任何地方被声明,但是只有在他们被声明以后才是合法有效的。因此,如果你在一个方法的开始定义了一个变量,那么它对于在该方法以内的所有程序都是可用的。反之,如果你在一个程序块的末尾声明了一个变量,它就没有任何用处,因为没有程序会访问它。例如,下面这个程序段就是无效的,因为变量count 在它被定义以前是不能被使用的。

 

    另一个需要记住的重要之处是:变量在其作用域内被创建,离开其作用域时被撤消。
这意味着一个变量一旦离开它的作用域,将不再保存它的值了。因此,在一个方法内定义的变量在几次调用该方法之间将不再保存它们的值。同样,在块内定义的变量在离开该块时也将丢弃它的值。因此,一个变量的生存期就被限定在它的作用域中。

     如果一个声明定义包括一个初始化,那么每次进入声明它的程序块时,该变量都要被重新初始化。例如,考虑这个程序:

 

// Demonstrate lifetime of a variable.
class LifeTime {
public static void main(String args[]) {
int x;
for(x = 0; x < 3; x++) {
int y = -1; // y is initialized each time block is entered
System.out.println("y is: " + y); // this always prints -1
y = 100;
System.out.println("y is now: " + y);
}
}
}

 

 

该程序运行的输出如下:

y is: -1
y is now: 100
y is: -1
y is now: 100
y is: -1
y is now: 100

 

    可以看到,每次进入内部的for循环,y都要被重新初始化为-1。即使它随后被赋值为100,该值还是被丢弃了。

    最后一点:尽管程序块能被嵌套,你不能将内部作用域声明的变量与其外部作用域声明的变量重名。在这一点上,Java 不同于C和C++。下面的例子企图为两个独立的变量起同样的名字。在Java 中,这是不合法的。但在C/C++ 中,它将是合法的,而且2个变量bar将是独立的。

 

// This program will not compile
class ScopeErr {
public static void main(String args[]) {
int bar = 1;
{
// creates a new scope
int bar = 2;
// Compile-time error – bar already defined!
}
}
}

 

     但是我们在现实中编写代码的时候为了使代码清楚明了,一般把方法的返回值赋值给一个变量,然后在执行后面的方法或者其他语句什么的。但是,假如这这个变量定义的作用域不对时,可能与你想要的结果风马牛不相及。然后你就开始怀疑你的逻辑是否正确。最后一步一步就陷入了僵局,无从下手。下面一谢宾斯三角形为例,简要介绍变量作用域和生存期的重要性:

 

错误的代码,定义的是全局变量

 

/**
 * 定义画三角形的方法
 * @param g:传入的的画布对象
 * @param aX:第一个点的横坐标
 * @param aY:第一个点的纵坐标
 * @param bX:第二个点的横坐标
 * @param bY:第二个点的纵坐标
 * @param cX:第三个点的横坐标
 * @param cY:第三个点的纵坐标
 */
public void DrawTri(Graphics g,int aX,int aY,int bX, int bY,int cX,int cY){
	g.drawLine(aX, aY, bX, bY);
	g.drawLine(aX, aY, cX, cY);
	g.drawLine(bX, bY, cX, cY);
}

/**
 * 定义谢宾斯基三角形递归方法
 * @param g:传入的画布对象
 * @param a1:第一个点的横坐标
 * @param a2:第二个点的纵坐标
 * @param b1:第二个点的横坐标
 * @param b2:第二个点的纵坐标
 * @param c1:第三个点的横坐标
 * @param c2:第三个点的纵坐标
 * @param count:递归次数计数器
 * @return	计数器
 */

int abX;
int abY;
int acX;
int acY;
int bcX;
int bcY;
public int Sierpinski(Graphics g,int a1,int a2,int b1, int b2,int c1,int c2,int count){
	abX = (a1 + b1)/2;
	abY = (a2 + b2)/2;
	acX = (a1 + c1)/2;
	acY = (a2 + c2)/2;
	bcX = (b1 + c1)/2;
	bcY = (b2 + c2)/2;
	DrawTri(g,abX,abY,acX, acY, bcX, bcY);
	
			
	if (count > 0){
	Sierpinski(g,abX,abY,acX, acY,a1, a2,count-1);
	Sierpinski(g,abX,abY,bcX, bcY,b1, b2,count-1);
	Sierpinski(g,acX,acY,bcX, bcY,c1, c2,count-1);
	}
	return count;
}
	
 

 

 运行结果就是 

 
  

 

下面是正确的代码,定义的局部变量 

/**
 * 定义画三角形的方法
 * @param g:传入的的画布对象
 * @param aX:第一个点的横坐标
 * @param aY:第一个点的纵坐标
 * @param bX:第二个点的横坐标
 * @param bY:第二个点的纵坐标
 * @param cX:第三个点的横坐标
 * @param cY:第三个点的纵坐标
 */
public void DrawTri(Graphics g,int aX,int aY,int bX, int bY,int cX,int cY){
	g.drawLine(aX, aY, bX, bY);
	g.drawLine(aX, aY, cX, cY);
	g.drawLine(bX, bY, cX, cY);
}

/**
 * 定义谢宾斯基三角形递归方法
 * @param g:传入的画布对象
 * @param a1:第一个点的横坐标
 * @param a2:第二个点的纵坐标
 * @param b1:第二个点的横坐标
 * @param b2:第二个点的纵坐标
 * @param c1:第三个点的横坐标
 * @param c2:第三个点的纵坐标
 * @param count:递归次数计数器
 * @return	count
 */
public int Sierpinski(Graphics g,int a1,int a2,int b1, int b2,int c1,int c2,int count){
	int abX = (a1 + b1)/2;
	int abY = (a2 + b2)/2;
	int acX = (a1 + c1)/2;
	int acY = (a2 + c2)/2;
	int bcX = (b1 + c1)/2;
	int bcY = (b2 + c2)/2;
	DrawTri(g,abX,abY,acX, acY, bcX, bcY);
	
			
	if (count > 0){
	Sierpinski(g,abX,abY,acX, acY,a1, a2,count-1);
	Sierpinski(g,abX,abY,bcX, bcY,b1, b2,count-1);
	Sierpinski(g,acX,acY,bcX, bcY,c1, c2,count-1);
	}
	return count;
}
	

 

     为什么前者得不到我们想到得到的图形,原因就是把局部变量定义为全局变量了,并不是逻辑不对,而是变量只需要在方法内作用,定义为全局变量就会产生在下一次循环时变量的值还不能自动注销,沿用上一次循环的值,不但会使程序得不到预期的结果,而且会影响程序的效率和增加计算机的内存占有。
     所以,在程序开发是一定要注意变量的生存期和作用域,在保证程序正常运行是尽量提高其高效,安全性。
  • 大小: 25.2 KB
分享到:
评论

相关推荐

    【大数据课设】p105出租车数据可视化分析-大数据-实训大作业.zip

    项目资源包含:可运行源码+数据集+文档 python + numpy, pandas, matplotlib, pyecharts, wordcloud 适用人群:学习不同技术领域的小白或进阶学习者;可作为课程设计、大作业、工程实训或初期项目立项。 数据来源:数据集taxis.csv从网络下载 数据清洗:异常值与缺失值的处理:有一些数据distance(乘车距离)为零而且上下车地点为空,还有些一些数据的payment(支付方式)为空。 数据预处理:将列名更改成中文 标准化与归一化: 数据分析: 数据可视化:

    TypeScript 入门教程

    TypeScript 入门教程

    人脸识别_课堂考勤_OpenCV_服务端系统_1741777828.zip

    人脸识别项目实战

    历届电赛试题及综合测评(真题+模拟题)

    本资源汇总了 历届全国电子设计竞赛(电赛)真题+模拟题,涵盖 电路设计、嵌入式系统、信号处理、自动控制等核心考点,并提供详细解析及综合测评,帮助参赛者高效备赛、查漏补缺、提升实战能力。 适用人群: 适合 准备参加电子设计竞赛的大学生、电赛爱好者、电子信息类相关专业的学生,以及希望提高电子设计和电路分析能力的工程师。 能学到什么: 电赛考察重点:熟悉往届竞赛的命题方向及考核重点。 电路设计与仿真:提升模拟电路、数字电路、单片机等核心技能。 问题分析与解决能力:通过综合测评找到薄弱点并针对性提升。 实战经验:掌握竞赛策略,提高应试效率和设计能力。 阅读建议: 建议先 通读真题,了解题型与解题思路,然后 结合模拟题实战演练,查找不足并通过测评强化练习,逐步提升竞赛能力。

    2024人工智能如何塑造未来产业:AI对各行业组织带来的的变革研究研究报告.pdf

    2024人工智能如何塑造未来产业:AI对各行业组织带来的的变革研究研究报告.pdf

    人脸识别_Golang_SDK_命令行登录_微信小程序应用_1741772240.zip

    人脸识别项目源码实战

    Vulkan原理与实战课程

    给大家分享一套课程——Vulkan原理与实战课程

    SiriYXR_Sokoban11_1741860914.zip

    c语言学习

    海豚鲸鱼数据集 5435张图 正确识别率可达92.6% 可识别:海豚 虎鲸 蜥蜴 海豹 鲨鱼 龟 支持yolov8格式标注

    海豚鲸鱼数据集 5435张图 正确识别率可达92.6% 可识别:海豚 虎鲸 蜥蜴 海豹 鲨鱼 龟 支持yolov8格式标注

    答谢中书书教学设计.docx

    答谢中书书教学设计.docx

    人脸识别_环境搭建_dlib_face_recognitio_1741771308.zip

    人脸识别项目源码实战

    网络技术_Web服务器_C语言_学习交流版_1741863251.zip

    c语言学习

    安卓开发_Gradle配置_React_Native_Meg_1741777287.zip

    人脸识别项目源码实战

    人工智能_深度学习_图像识别_UI界面_项目展示.zip

    人脸识别项目实战

    基于Springboot框架的美发门店管理系统的设计与实现(Java项目编程实战+完整源码+毕设文档+sql文件+学习练手好项目).zip

    本美发门店管理系统有管理员和用户两个角色。用户功能有项目预定管理,产品购买管理,会员充值管理,余额查询管理。管理员功能有个人中心,用户管理,美容项目管理,项目类型管理,项目预定管理,产品库存管理,产品购买管理,产品入库管理,会员卡管理,会员充值管理,余额查询管理,产品类型管理,系统管理等。因而具有一定的实用性。 本站是一个B/S模式系统,采用SSM框架,MYSQL数据库设计开发,充分保证系统的稳定性。系统具有界面清晰、操作简单,功能齐全的特点,使得美发门店管理系统管理工作系统化、规范化。本系统的使用使管理人员从繁重的工作中解脱出来,实现无纸化办公,能够有效的提高美发门店管理系统管理效率。 关键词:美发门店管理系统;SSM框架;MYSQL数据库;Spring Boot 1系统概述 1 1.1 研究背景 1 1.2研究目的 1 1.3系统设计思想 1 2相关技术 2 2.1 MYSQL数据库 2 2.2 B/S结构 3 2.3 Spring Boot框架简介 4 3系统分析 4 3.1可行性分析 4 3.1.1技术可行性 4 3.1.2经济可行性 5 3.1.3操作可行性 5 3.2系

    Python实现基于SSA-CNN-GRU麻雀算法优化卷积门控循环单元数据分类预测的详细项目实例(含完整的程序,GUI设计和代码详解)

    内容概要:本文档介绍了基于SSA-CNN-GRU麻雀算法优化卷积门控循环单元数据分类预测的详细项目实例,重点讲述了该项目的背景、目标、挑战与解决方案、技术特点、应用领域等方面的内容。文档详细记录了从项目启动、数据预处理、算法设计(SSA优化CNN-GRU模型)、构建与评估模型到实现美观的GUI界面整个过程,并讨论了防止过拟合的技术如正则化、早停和超参数优化。另外还涵盖了项目扩展的可能性、部署和应用策略、需要注意的地方以及未来改进的方向。全文强调了模型的泛化能力和计算效率,展示了该混合算法模型在实际应用中的优越性能。 适合人群:具备一定的Python编程经验及机器学习基础知识的研究人员和技术人员;对深度学习、智能优化算法及实际应用感兴趣的学者和从业者;寻求提升数据分析和预测准确性的金融分析师、数据科学家等相关专业人士。 使用场景及目标:本文档非常适合用作学习和参考资料,以掌握如何将SSA、CNN与GRU三种先进技术结合起来进行复杂的分类和预测问题求解。具体应用场景包括但不限于以下几个方面:金融领域——股票价格预测;医疗保健领域——辅助诊断;工业制造——预防性维护;智能家居——个性化服务;以及其他涉及到时序数据分析和多模态数据处理的场合。文档既包含了理论知识又提供了完整的源代码示例,可以帮助读者理解算法原理并通过实践中加深对其的认识。 其他说明:该项目不仅仅是关于算法的设计实现,更是有关于系统的整体架构规划以及工程上的考量,比如环境准备(确保环境洁净、必要包的安装等)、数据准备、GPU配置支持等等。同时文中给出了详细的代码片段,方便开发者理解和复现实验成果。值得注意的是,虽然文中提供了一套通用解决方案,但在真实场景下还需要针对性的调整参数或修改网络结构来达到最好的性能效果。此外,对于追求更高的预测精度或解决更大规模的问题,作者建议进一步探索深度强化学习等高级技术和多任务学习策略,并且考虑使用增量学习让模型能够适应新数据而不必重新训练整个模型。最后提到安全性和隐私保护也是项目实施过程中的重要因素,要妥善保管用户的敏感信息并且做到合法合规地收集和使用数据。

    人脸识别_T形分布_Gabor变换_特征提取_增强鲁棒性_1741777397.zip

    人脸识别项目实战

    13005463562_FaceWeb_1741771809.zip

    人脸识别项目实战

    水下垃圾检测数据集,基于voc和yolo标注的两种格式

    水下垃圾检测数据集,基于voc和yolo标注的两种格式,共23,056个文件,已经划分了训练集和验证集、测试集。并且提供了真实水下的视频数据,可以用作视频推理

    (参考GUI)MATLAB车辆检测.zip

    (参考GUI)MATLAB车辆检测.zip

Global site tag (gtag.js) - Google Analytics