大多数程序设计语言都提供了“作用域”(scope)的概念。
对于在作用域里定义的名字,作用域同时决定了它的“可见性”以及“存在时间”。在c,c++和java里,作用域是由花括号的位置决定的。
变量的作用域
例子:
1
2
3
4
5
6
7
8
9
10
|
{
int x = 12 ;
/* only x available */
{
int q = 96;
/* both x & q available */
}
/* only x available */
/* q “out of scope” */
}
|
java用一对大括号作为语句块的范围,称为作用域,作为在作用域里定义的一个变量,它只有在哪个作用域结束之前才可使用。
在java中不能像下面这样书写代码:
1
2
3
4
5
6
|
{
int x = 12 ;
{
int x = 96 ; /* illegal */
}
}
|
java编译器会认为变量已被定义,所以作用域中的变量不能重复定义,但是在c和c++中能将一个变量“隐藏”在一个更大的作用域里,在c和c++中被允许,在java中是不允许的,因为java的设计者认为这样做使程序产生了混淆。
再来看两个例子,为了让大家看到效果这里使用截图的方式:
①
②
我们再来看这段代码,大家思考一个问题,第11行和第12行为什么没有语法错误?代码如下:
如果我们交换一下位置呢,代码如下:
离开作用域,变量所分配的内存空间将被jvm回收,所以语法不会有错误,而第二种写法name并没有离开{}作用域,所以会语法错误。
上述的变量都是局部变量,那么如果是在有全局变量的情况下又是怎样一种结果呢?我们来用代码说话,代码如下:
大家仔细的观察并结合代码思考,可以得出变量的作用域结论如下:
在同一作用域范围的包裹下全局变量名和局部变量名是可以变量名相同的,在同一个作用域范围的包裹下局部变量和局部变量不可以变量名相同(作用域内不能重复命名),使用变量的时候如果不指明使用全局变量还是局部变量,那么默认的就是使用局部的那个变量,但是如果局部变量超出了它本身的作用域范围则会失效,被jvm垃圾回收,那么则可以重复命名此变量,并使用最新定义的这个局部变量。
对象的作用域
java对象不具备与朱类型一样的存在时间。用new关键字创建一个java对象的时候,它会超出作用域的范围之外。所以假若使用下面这段代码:
1
2
3
|
{
string s = new string( "a string" );
} /* 作用域的终点 */
|
那么句柄s,也就是引用会在作用域的终点处消失。然而,s指向的string对象依然占据着内存空间。在上面这段代码里,我们没有办法继续使用这个对象,因为指向它的唯一一个句柄已经超出了作用域的边界。
这样造成的结果是:对于用new创建的对象,只要我们愿意,它们就会一直保留下去。这个编程问题在c和c++里特别突出。在c++里遇到的麻烦最大:由于不能从语言获得任何帮助,所以在需要对象的时候,根本无法确定它们是否可用。而且最麻烦的是,在c++里,一旦完成工作,必须保证将对象手动清除。
这样便带来了一个有趣的问题。假如 java 让对象依然故我,怎样才能防止它们大量充斥内存,并最终造成程序的“凝固”呢。在 c++里,这个问题最令程序员头痛。但 java 以后,情况却发生了改观。 java 有一个特别的“垃圾收集器”,它会查找用 new 创建的所有对象,并辨别其中哪些不再被引用。随后,它会自动释放由那些闲置对象占据的内存,以便能由新对象使用。这意味着我们根本不必操心内存的回收问题。只需简单地创建对象,一旦不再需要它们,它们就会自动离去。这样做可防止在 c++里很常见的一个编程问题:由于程序员忘记释放内存造成的“内存溢出”。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持服务器之家!
原文链接:http://www.cnblogs.com/AlanLee/p/6627949.html