《Java编程思想》(第二版)第4章:初始化和清理

发表于:2007-07-04来源:作者:点击数: 标签:
每个函数的第一个字母以小写表示这种编程风格不适用于构造函数身上,因为构造函数的名称必须完全吻合class名称 在Java(和C++)中,构造函数是函数名称必须能被重载的另一个原因。每个重载函数必须拥有独一无二的引数列(argument list),而不可以以返回值(r
每个函数的第一个字母以小写表示"这种编程风格不适用于构造函数身上,因为构造函数的名称必须完全吻合class名称

在Java(和C++)中,构造函数是"函数名称必须能被重载"的另一个原因。每个重载函数必须拥有独一无二的引数列(argument list),而不可以以返回值(return value)作为重载的基准,因为有时候我们是以f()来执行函数调用的,但是此时并没有返回值的任何信息,编译器会束手无策。

Default构造函数会不会被编译器主动提供则类似于C++:当您没有定义任何构造函数的时候编译器会自动提供,而当您提供了任何一个构造函数的时候,编译器是不会再给你提供default构造函数的。

有时候为了方便我们需要在构造函数中调用构造函数。下面的例子告诉我们一些行为规则:

// Calling constructors with "this"

public class Flower{
int petalCount=0;
String s=new String("null");
Flower(int petals){
petalCount=petals;
}
Flower(String ss){
s=ss;
}
Flower(String s,int petals){
this(petals);
//! this(s); //Can''t call twice!
this.s=s;
}
Flower(){
this("hi",47);
}
void print(){
//! this (11); //Not inside non-constructor!
}
public static void main(String[] args){
Flower x=new Flower();
x.print();
}
}

此段程序告诉我们几点:
1.虽然你能够藉由this调用一个构造函数,却不能以相同手法调用两个。
2.对另一构造函数的调用动作必须置于最其实除,否则编译器会发出错误消息。
3.在print()函数中你可以看出,编译器不允许你在构造函数以外的任何函数内调用构造函数,即构造函数只用于new关键字由系统自动调用。


finalize()第一种用法可以在垃圾回收之前给你做一些特殊的处理,因为垃圾回收只回收内存的,如果你有一些别的资源应该释放或者在对象销毁之前作一些动作比如说擦除以前绘制的图像等等,就应该会用到它。第二种用法:当采用原生函数(native methods)时,可以在finalize中调用free()来释放先前malloc()函数分配的内存。

Garbage collector的运作依赖于Java的两个关键技术:单根继承技术(继承自Object),对象完全从堆(heap)中分配内存(而非类成员的基本类型变量则放在stack中分配,为了效率的考虑)

在Java中 定义和初始化是一体的,两者不可能彼此脱离而独立存在。

成员初始化:Java保证,变量绝对会在被使用之前被适当初始化。当变量被定义与函数之内时,Java会运用编译期错误消息来贯彻它的保证。如果某个class的数据成员隶属基本类型,情况则稍有不同。因为所有函数都可以初始化或使用该数据值,所以强迫使用者一定得在使用之前给定适当初始值是不切实际的。不过,放任它持有毫无意义的处置也很危险,所以每个"隶属基本类别"的class数据成员都保证一定有初始值,即编译器设定默认初始值了。当然我们也可以方便的直接在class的变量定义出指定其值。构造函数可用来执行初始化操作,此时class中的初始化次序取决于变量在class中的定义次序。对于static的初始化动作之在必要时(一次且只有一次)可才会发生(比如被访问。或者所属的class第一个对象被建立此时,此时该static变量先于所有non-static变量初始化),自此之后,static对象便不会再被初始化。

Array的初始化:任何数组对象都隐含了length成员。在对数据元素存取的时候要做Java会对执行边界检查,而且你无法关闭此功能,牺牲程序的执行效率但是更安全,消除了C/C++数组的弊端,还完全灭杀了buffer overflow 攻击者的生存空间。

原文转自:http://www.ltesting.net