Java编程思想学习笔记五:初始化与清理

一、用构造器确保初始化

    Java中通过提供构造器,确保每个类的对象都可以得到初始化,构造器的形式为:
className(){
    //---
}

在类的内部定义的一个与类名相同的方法,该方法没有返回值,没有返回值并不是返回void,而是真正的无返回值。该方法在对象创建时会自动执行。

package com.chenxyt.java.practice;
public class ConstructorTest{
	ConstructorTest(){
		System.out.println("Constructor Begin");
	}
	public static void main(String[] args) {
		for(int i=0;i<5;i++){
			ConstructorTest ct = new ConstructorTest();
		}
	}
}

运行结果如下:

    
可以看见程序在初始化对象的时候自动执行了构造方法。
    上面是无参的构造方法,还可以显示的编写有参的构造方法,给类的成员变量赋值:
package com.chenxyt.java.practice;
public class ConstructorTest{
	private int age;
	ConstructorTest(int i){
		age=i;
	}
	public static void main(String[] args) {
		for(int i=0;i<5;i++){
			ConstructorTest ct = new ConstructorTest(i);
			System.out.println("Age=" + ct.age);
		}
	}
}

运行结果:

    
如果类中只有唯一的一个带参数的构造器,那么默认的无参构造器将不可用。

二、方法重载

    接上一章,假如我们有多种多样的需求,想要初始化不同形态的对象,这时候我们可能需要多个构造器才能满足需求。而构造器的名字又是固定的,所以这个时候变化的就是参数类型或者是参数个数。我们把这种方法名不变,只改变参数类型或者参数个数的操作叫做方法重载。方法重载不仅支持构造器方法重载,也支持普通方法的重载。
package com.chenxyt.java.practice;
public class ConstructorTest{
	private int age;
	private String name;
	ConstructorTest(){
		
	}
	ConstructorTest(int i){
		age=i;
	}
	ConstructorTest(String j){
		name=j;
	}
	ConstructorTest(int i,String j){
		age=i;
		name=j;
	}
	public static void main(String[] args) {
		ConstructorTest ct1 = new ConstructorTest();
		ConstructorTest ct2 = new ConstructorTest(22);
		ConstructorTest ct3 = new ConstructorTest("张三");
		ConstructorTest ct4 = new ConstructorTest(23,"李四");
		System.out.println("ct1 age=" + ct1.age + "---name=" + ct1.name);
		System.out.println("ct2 age=" + ct2.age + "---name=" + ct2.name);
		System.out.println("ct3 age=" + ct3.age + "---name=" + ct3.name);
		System.out.println("ct4 age=" + ct4.age + "---name=" + ct4.name);
	}
}

运行结果如下:

    
    四个不同的构造函数初始化了四个不同的对象,可以从打印结果看出,没有初始化的值int类型为0,String类型为null。
package com.chenxyt.java.practice;
public class ConstructorTest{
	private static void print(int i){
		System.out.println("int 类型打印" + i);
	}
	private static void print(String j){
		System.out.println("String 类型打印" + j);
	}
	public static void main(String[] args) {
		ConstructorTest.print(22);
		ConstructorTest.print("张三");
	}
}

    普通方法的重载最明显的使用就是println()方法,我们要保证打印任何内容都使用println()方法,而不是打印int类型用一个方法printint(),然后打印String类型用一个方法printString(),这时候方法重载是最好的办法。

    
    如上结果显示了两个重载方法的不同结果。
    重载方法中如果传入了低类型的参数,那么如果找不到合适的方法,会被隐式的提升成高类型的数据。如果传入了高类型的参数,如果不进行类型转换,那么编译器会报错。Java中区分重载方法的依据是参数类型和参数个数,参数顺序区分会造成程序易读性较差,在一些情况下,也可以使用返回值区分,当然前提是如果你关心方法的返回值。

三、缺省构造器

    如前文所述,默认的构造器,就是没有形参的构造器,作为一个类的缺省构造器。如果程序员没有显示的在代码中创建一个构造器,那么Java会自动帮你创建一个无参构造器来完成初始化。当然如果程序员显示的创建了构造函数,那么Java就不会给你创建缺省构造器了。

四、this关键字

    this关键字也是Java中尤为重要的一个关键字,主要有三个作用:
1.表示当前对象的引用,并返回当前对象,用法如下:
package com.chenxyt.java.practice;
public class ThisTest{
	public ThisTest doFunc(){
		return this;
	}
	public static void main(String[] args) {
		ThisTest tt = new ThisTest().doFunc();
	}
}

doFunc()方法返回的是一个ThisTest类型的值,所以此处使用return this,返回了该类型对象的引用。

2.表示类的成员变量,在有的构造函数中,形参与成员变量使用同一个字符串,这时候可以用this来区分,带有this的表示成员变量,用法如下:
package com.chenxyt.java.practice;
public class ThisTest{
	private String arg;
	ThisTest(String arg){
		this.arg = arg;
	}
	public static void main(String[] args) {
		ThisTest tt = new ThisTest("嘻嘻");
		System.out.println(tt.arg);
	}
}

运行结果:

    
此处this.arg表明了这个变量是成员变量,与构造方法的形参做了区分。
3.可以在构造器中调用另一个构造器,使用this带参数代替构造器的方法名,用法如下:
package com.chenxyt.java.practice;
public class ThisTest{
	ThisTest(int i,String j){
		this(j);
		System.out.println("我是构造器1");
	}
	ThisTest(String arg){
		System.out.println("我是构造器2");
	}
	public static void main(String[] args) {
		ThisTest tt = new ThisTest(2,"嘻嘻");
	}
}

运行结果如下:

    
可以看到对象在初始化的时候使用了两个构造器,其中一个构造器是在另一个构造器中通过this()的方式进行了调用。

五、清理:终结处理和垃圾回收

    Java中的内存清理使用的是Java自带的垃圾回收机制,但是不管怎么来说,这种方式都不是绝对安全的。Java的垃圾回收机制清理的是通过new创建的对象,而在某些特殊情况下,可能有些对象不是通过new创建的,这些对象如果不使用的时候,垃圾回收器是不能准确的清理他们的从而造成了这块特殊的内存区域一直得不到释放。Java中提供了finalize()方法来处理这一部分特殊的内存区域。它的处理流程是这样的,在Java垃圾回收器启用之前,会先调用这个方法进行一些必要的回收操作。但是针对这一块特殊的区域,或者是new创建的需要被回收的对象,一般情况下只有当Java虚拟机内存快要消耗殆尽的时候,垃圾回收器才会启动,毕竟启动垃圾回收器也是需要消耗资源的,所以不可能说实时存在。所说的特殊的创建对象方式,一般是指“本地方法”使用时,也就是在Java中调用非Java代码的时候发生的。所以一般情况下是不需要使用finalize()方法的。
    finalize()通常还有另一个用法,由于它是在Java垃圾回收器启动之前执行的,所以可以用它来判断终结状态,即判断一个对象是否满足回收条件。

六、成员初始化

    Java尽量保证每个变量在使用之前都进行了初始化操作,变量分为局部变量和成员变量,局部变量如果没有显示的初始化,在使用它的时候会报错,而成员变量不会,如果没有显示的初始化一个成员变量,那么它会被默认的分配一个指定的值。

七、构造器初始化

    如前边对构造器的阐述,可以使用构造器来初始化类的成员变量,当对象被实例化之后,对象的成员变量会被初始化。静态成员变量只有在第一次使用它是才会被初始化,后边再次用到时不会被初始化。初始化顺序为创建对象时,先初始化这个类的静态变量,然后在堆上为这个对象分配内存,最后执行构造函数。

八、数组初始化

    数据是一系列相同数据类型封装起来的序列,它的初始化可以发生在任何时候,int[] a1表示一个int类型的数组,这个数组内部所有的值都是int类型,a1只是这个数组的一个引用,可以显示的通过如int[] a1={1,2,3};的形式进行初始化。如果不能确定数组的内容或者是长度,则可以通过new的形式来创建一个数组。int[] a = new int[20];这种创建也只是创建了一个引用数组,直到数组中的每一个字段都有确切的值了,初始化才真正的完成。如a[1]=3;
   使用数组我们可以构建一个变参的函数,就是当方法的参数类型和个数都不确定的时候,我们可以使用一个数组作为形参。因为Object类是所有类型的父类,所以这个形参数组的类型就是Obejct类,对于基本数据类型,因为都有对应的包装类,所以也可以转换成Obejct类进行使用。
package com.chenxyt.java.practice;
public class DifArgTest{
	public static void printArray(Object[] args){
		for(Object obj:args){
			System.out.print(obj + "");
		}
	}
	public static void main(String[] args) {
		DifArgTest.printArray(new Object[]{"我今年",new Integer(24),"岁!"});
	}
}
运行结果:
    
可以看到printArray()方法将Obejct的数组内容都打印了出来。但是这样传参数是要写成数组的形式,未免有些臃肿。JDK1.5之后真正的变参函数出来啦。
package com.chenxyt.java.practice;
public class DifArgTest{
	public static void printArray(Object...args){
		for(Object obj:args){
			System.out.print(obj + "");
		}
	}
	public static void main(String[] args) {
		DifArgTest.printArray("我","今年",24,"岁");
	}
}
用“...”代替了原有的数组符号,同时传递的参数也变得更加简洁了。运行结果:
    

九、枚举类型

    这里只是初步的了解一下枚举类型,enum,Java中的enum要比C++更加完备。以下是一个简单的例子:
package com.chenxyt.java.practice;
public class EnumTest{
	public enum EnumSet{
		FIRST,SECOND,THIRD
	}
	public static void main(String[] args) {
		EnumSet es = EnumSet.FIRST;
		System.out.println(es);
	}	
}
创建一个枚举类型,类型内部的实例值是常量,因此按照通用的命名规范进行大写。同时需要使用枚举时,可以初始化一个引用然后进行赋值。结果如下:
    
    当我们创建枚举的时候编译器会自动为我们添加一些有用的特性,我觉得这是与其它语言相比更加完备的地方,比如它会创建一个toString()方法,这也就是为什么我们上边可以使用syso打印出来。编译器还会创建ordinal()方法,用来表示特定enum常量的声明顺序,以及一个static values()方法,该方法是一个静态的方法可以通过enum名字进行访问,方法的作用是按照enmu的声明顺序,产生一个由enum常量值组成的数组。
package com.chenxyt.java.practice;
public class EnumTest{
	public enum EnumSet{
		FIRST,SECOND,THIRD
	}
	public static void main(String[] args) {
		for(EnumSet et:EnumSet.values()){
			System.out.println("value is:" + et + "---ordinal is:" + et.ordinal());
		}
	}	
}

运行结果如下:
    
    euum看起来像是一种新的数据类型,但是实际上enum是一个类,并且具有自己的方法。
    除了上边的特性之外,enum还有个更加实用的特性,由于它是一个常量集,因此可以和switch语句完美匹配。
package com.chenxyt.java.practice;
public class EnumTest{
	public EnumSet es;
	EnumTest(EnumSet es){
		this.es = es;
	}
	public enum EnumSet{
		FIRST,SECOND,THIRD
	}
	public void doSwitchPrint(){
		switch(es){
		case FIRST:
			System.out.println("This is First");
			break;
		case SECOND:
			System.out.println("This is Second");
			break;
		case THIRD:
			System.out.println("This is Third");
			break;
		default:
			System.out.println("This is Error");
		}
	}
	public static void main(String[] args) {
		EnumTest et1 = new EnumTest(EnumSet.FIRST);
		EnumTest et2 = new EnumTest(EnumSet.SECOND);
		EnumTest et3 = new EnumTest(EnumSet.THIRD);
		et1.doSwitchPrint();
		et2.doSwitchPrint();
		et3.doSwitchPrint();
	}	
}
运行结果如下:
    
    以上是对enum的一个初步了解,后续将会单独阐述。

十、总结

    本章在Java中占有了至关重要的作用,也可以说初始化比较重要。总的来说要掌握如何进行初始化,方法的重载,重载不仅发生在构造方法中,也可以发生在普通方法。this关键的使用可以说很重要,但是便于理解。垃圾回收以及枚举只是初步的了解了一下,后续还会单独进行学习。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值