`
什么世道
  • 浏览: 222852 次
  • 性别: 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
分享到:
评论

相关推荐

    YOLO算法-城市电杆数据集-496张图像带标签-电杆.zip

    YOLO系列算法目标检测数据集,包含标签,可以直接训练模型和验证测试,数据集已经划分好,包含数据集配置文件data.yaml,适用yolov5,yolov8,yolov9,yolov7,yolov10,yolo11算法; 包含两种标签格:yolo格式(txt文件)和voc格式(xml文件),分别保存在两个文件夹中,文件名末尾是部分类别名称; yolo格式:<class> <x_center> <y_center> <width> <height>, 其中: <class> 是目标的类别索引(从0开始)。 <x_center> 和 <y_center> 是目标框中心点的x和y坐标,这些坐标是相对于图像宽度和高度的比例值,范围在0到1之间。 <width> 和 <height> 是目标框的宽度和高度,也是相对于图像宽度和高度的比例值; 【注】可以下拉页面,在资源详情处查看标签具体内容;

    (177406840)JAVA图书管理系统毕业设计(源代码+论文).rar

    JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代码+论文) JAVA图书管理系统毕业设计(源代

    (35734838)信号与系统实验一实验报告

    内容来源于网络分享,如有侵权请联系我删除。另外如果没有积分的同学需要下载,请私信我。

    YOLO算法-椅子检测故障数据集-300张图像带标签.zip

    YOLO系列算法目标检测数据集,包含标签,可以直接训练模型和验证测试,数据集已经划分好,包含数据集配置文件data.yaml,适用yolov5,yolov8,yolov9,yolov7,yolov10,yolo11算法; 包含两种标签格:yolo格式(txt文件)和voc格式(xml文件),分别保存在两个文件夹中,文件名末尾是部分类别名称; yolo格式:<class> <x_center> <y_center> <width> <height>, 其中: <class> 是目标的类别索引(从0开始)。 <x_center> 和 <y_center> 是目标框中心点的x和y坐标,这些坐标是相对于图像宽度和高度的比例值,范围在0到1之间。 <width> 和 <height> 是目标框的宽度和高度,也是相对于图像宽度和高度的比例值; 【注】可以下拉页面,在资源详情处查看标签具体内容;

    基于小程序的新冠抗原自测平台小程序源代码(java+小程序+mysql+LW).zip

    系统可以提供信息显示和相应服务,其管理新冠抗原自测平台小程序信息,查看新冠抗原自测平台小程序信息,管理新冠抗原自测平台小程序。 项目包含完整前后端源码和数据库文件 环境说明: 开发语言:Java JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea Maven包:Maven3.3 部署容器:tomcat7 小程序开发工具:hbuildx/微信开发者工具

    YOLO算法-俯视视角草原绵羊检测数据集-4133张图像带标签-羊.zip

    YOLO系列算法目标检测数据集,包含标签,可以直接训练模型和验证测试,数据集已经划分好,包含数据集配置文件data.yaml,适用yolov5,yolov8,yolov9,yolov7,yolov10,yolo11算法; 包含两种标签格:yolo格式(txt文件)和voc格式(xml文件),分别保存在两个文件夹中,文件名末尾是部分类别名称; yolo格式:<class> <x_center> <y_center> <width> <height>, 其中: <class> 是目标的类别索引(从0开始)。 <x_center> 和 <y_center> 是目标框中心点的x和y坐标,这些坐标是相对于图像宽度和高度的比例值,范围在0到1之间。 <width> 和 <height> 是目标框的宽度和高度,也是相对于图像宽度和高度的比例值; 【注】可以下拉页面,在资源详情处查看标签具体内容;

    (171674830)PYQT5+openCV项目实战:微循环仪图片、视频记录和人工对比软件源码

    内容来源于网络分享,如有侵权请联系我删除。另外如果没有积分的同学需要下载,请私信我。

    新建 文本文档.docx

    新建 文本文档.docx

    hw06.zip

    hw06

    3. Kafka入门-安装与基本命令

    3. Kafka入门-安装与基本命令

    燃气管道施工资质和特种设备安装改造维修委托函.docx

    燃气管道施工资质和特种设备安装改造维修委托函.docx

    The state of AI 2024.pdf

    AI大模型研究相关报告

    lab02.zip

    lab02

    Unity视频插件AVPro的Win端2.2.3

    仅供学习使用,其他用途请购买正版资源AVPro Video Core Windows Edition 2.2.3 亲测可用的视频播放插件,能丝滑播放透明视频等.

    建设工程消防验收现场指导意见表.docx

    建设工程消防验收现场指导意见表.docx

    MVIMG_20241222_194113.jpg

    MVIMG_20241222_194113.jpg

    五相电机双闭环矢量控制模型-采用邻近四矢量SVPWM-MATLAB-Simulink仿真模型包括: (1)原理说明文档(重要):包括扇区判断、矢量作用时间计算、矢量作用顺序及切时间计算、PWM波的生成

    五相电机双闭环矢量控制模型_采用邻近四矢量SVPWM_MATLAB_Simulink仿真模型包括: (1)原理说明文档(重要):包括扇区判断、矢量作用时间计算、矢量作用顺序及切时间计算、PWM波的生成; (2)输出部分仿真波形及仿真说明文档; (3)完整版仿真模型:包括邻近四矢量SVPWM模型和完整双闭环矢量控制Simulink模型; 资料介绍过程十分详细,零基础手把手教学,资料已经写的很清楚

    YOLO算法-锡罐-牙罐-盖子打开数据集-179张图像带标签-锡罐-牙罐-盖子打开.zip

    YOLO系列算法目标检测数据集,包含标签,可以直接训练模型和验证测试,数据集已经划分好,包含数据集配置文件data.yaml,适用yolov5,yolov8,yolov9,yolov7,yolov10,yolo11算法; 包含两种标签格:yolo格式(txt文件)和voc格式(xml文件),分别保存在两个文件夹中,文件名末尾是部分类别名称; yolo格式:<class> <x_center> <y_center> <width> <height>, 其中: <class> 是目标的类别索引(从0开始)。 <x_center> 和 <y_center> 是目标框中心点的x和y坐标,这些坐标是相对于图像宽度和高度的比例值,范围在0到1之间。 <width> 和 <height> 是目标框的宽度和高度,也是相对于图像宽度和高度的比例值; 【注】可以下拉页面,在资源详情处查看标签具体内容;

    java毕设项目之ssm基于JSP的乡镇自来水收费系统+jsp(完整前后端+说明文档+mysql+lw).zip

    项目包含完整前后端源码和数据库文件 环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea Maven包:Maven3.3 服务器:tomcat7

Global site tag (gtag.js) - Google Analytics